C++ 探索类成员变量的初始化

普通的成员变量

#include <string>
#include <iostream>
using namespace std;

class CTest {
public:
	int a,c;
	string s;
	CTest() { cout << "create CTest.\n"; a = 1; s = "a"; }
	~CTest(){ cout << "destory CTest.\n";}
	int Get() { return b; }
	string GetSS() { return ss; }
private:
	int b;
	string ss;
};
CTest g_test;

int main()
{
	CTest test;
	cout << test.a << "\t" << test.s << "\t"<<test.Get()<<"\t"<<test.GetSS()<<"\t"<<test.c<<endl;
	cout << g_test.a << "\t" << g_test.s << "\t" << g_test.Get() << "\t" << g_test.GetSS() << "\t"<<g_test.c<<endl;
	return 0;
}

以上实验说明:
全局变量实例其内置类型初始化为0,且调用默认的构造函数;
局部变量实例其内置类型没有初始化,值未知,也调用默认构造函数。

静态变量

#include <string>
#include <iostream>
using namespace std;

class CTest {
public:
	int a,c;
	string s;
	CTest() { cout << "create CTest.\nst :"<<st<<endl; a = 1; s = "a"; }
	~CTest(){ cout << "destory CTest.\nst :"<<st<<endl;}
	int Get() { return b; }
	string GetSS() { return ss; }
	static int st;
private:
	int b;
	string ss;
};
int CTest::st = 88;

CTest g_test;
void ChangeClassStaticVal()
{
	CTest::st = 8888;
}

int main()
{
	CTest::st = 666;
	CTest test;
	cout << test.a << "\t" << test.s << "\t"<<test.Get()<<"\t"<<test.GetSS()<<"\t"<<test.c<<"\t"<< CTest::st<<endl;
	ChangeClassStaticVal();
	cout << g_test.a << "\t" << g_test.s << "\t" << g_test.Get() << "\t" << g_test.GetSS() << "\t"<<g_test.c<<"\t"<< CTest::st<<endl;
	CTest::st = 666;
	return 0;
}

以上实验说明:
由于类的静态成员变量一定要在类外显示赋初始值,且只能初始化一次,故无论何种情况都以赋值的结果为准,但是在函数内可以多次修改静态变量值
如果没有在类外赋值,则报错:“无法解析的外部符号 “public: static int CTest::st” (?st@CTest@@2HA)”;
如果是多次在类外赋初值,则报错:“error C2374: “st”: 重定义;多次初始化”;

成员常量

#include <string>
#include <iostream>
using namespace std;

class CTest {
public:
	int a,c;
	string s;
	CTest():ci(6666) { cout << "create CTest.\nst :" << st << endl; a = 1; s = "a"; }
	~CTest(){ cout << "destory CTest.\nst :"<<st<<endl;}
	int Get() { return b; }
	string GetSS() { return ss; }
	static int st;
	const int ci;
	const static int csti;
	static const int stci;
private:
	int b;
	string ss;
};

int CTest::st = 88;
const int CTest::csti = 123456;
const int CTest::stci = 123456;

以上实验说明:
因为在程序进入构造函数时成员常量就不能再改变,所以必须在构造函数的初始化列表赋初值;
静态常量则要在类外初始化,且不能修改。

成员变量初始化顺序

参考:青雲-吾道乐途
总结:
1、成员变量在使用初始化列表初始化时,与构造函数中初始化成员列表的顺序无关,只与定义成员变量的顺序有关。因为成员变量的初始化次序是根据变量在内存中次序有关,而内存中的排列顺序早在编译期就根据变量的定义次序决定了。这点在EffectiveC++中有详细介绍。
2、如果不使用初始化列表初始化,在构造函数内初始化时,此时与成员变量在构造函数中的位置有关。
3、注意:类成员在定义时,是不能初始化的
4、注意:类中const成员常量必须在构造函数初始化列表中初始化。
5、注意:类中static成员变量,必须在类外初始化。
6、静态变量进行初始化顺序是基类的静态变量先初始化,然后是它的派生类。直到所有的静态变量都被初始化。这里需要注意全局变量和静态变量的初始化是不分次序的。这也不难理解,其实静态变量和全局变量都被放在公共内存区。可以把静态变量理解为带有“作用域”的全局变量。在一切初始化工作结束后,main函数会被调用,如果某个类的构造函数被执行,那么首先基类的成员变量会被初始化。


class CTest
{
public:
	int a = 1;
	CTest() {}
	CTest(int a_) :a(a_) {}
	CTest(int a_, bool b) :a(4) { a = a_; }
};

int main()
{
	CTest a1, a2(3), a3(5, true);
	cout << "a1.a=" << a1.a << endl;
	cout << "a2.a=" << a2.a << endl;
	cout << "a3.a=" << a3.a << endl;
	return 0;
}

关于成员变量在定义时就初始化的问题,这是c++11标准下是可以在定义时就初始化,之前则不能。
且初始化列表初始化的变量值会覆盖掉声明时初始化的值,而构造函数中初始化的值又会覆盖掉初始化列表的


因作者水平有限,难免有错误之处,欢迎读者指正!

上一篇:引用计数与强弱指针


下一篇:C++复习记录(一)