不要显式调用构造函数和析构函数

常用的代码如下:

1
2
Base b;
cout<<"hello"<<endl;

结果:
1
2
3
基类构造 0x62fe84
hello
基类析构0x62fe84

但是这样的代码就不同了:

1
2
Base();		// 匿名的临时对象
cout<<"hello"<<endl;

运行结果:
1
2
3
基类构造 0x62fe84
基类析构0x62fe84
hello

可见临时创建的类对象立刻销毁了,这与平时创建在stack上的对象再出了局部范围再销毁是不同的。

看这样的代码,是关于显式调用构造函数导致的成员变量未初始化问题:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
class CTest  
{
public:
CTest()
{
m_a = 1;
}
CTest(int b)
{
m_b = b;
CTest();
}
~CTest() {}
void showA()
{
std::cout<<"a:"<<m_a;
}
private:
int m_a;
int m_b;
};

void main()
{
CTest myTest(2);
t.showA();
}

结果是a:6487936,也就是m_a未初始化。这里我们创建的对象是myTest,希望对其成员m_a初始化,但在构造函数里显式调用另一个构造函数,实际上是创建了一个临时对象,这个对象对m_a初始化了,这跟myTest是没有关系的,而且它很快又销毁了,所以没有达到目的。

这样的代码也是有问题的:

1
2
3
Base b;
b.~Base();
cout<<"hello"<<endl;

结果是二次析构了:

1
2
3
4
基类构造 0x62fe84
基类析构 0x62fe84
hello
基类析构 0x62fe84

所以不要显式调用构造函数和析构函数,这是危险的。