现在有这么一个问题,有一个带参数的构造函数,当默认构造去调这个带参数的构造,在Java中大家都明白,直接穿个this(xxx)就可以了,那就在C++中模仿一下,写出下面代码:
class A {private: string name; int age;public: A(string name, int age) { this->name = name; this->age = age; } A() { A("bob", 20); } void print() { cout << name << " " << age << endl; } // ... // ...};当调用如下:
A a;a.print();输出:
125156784一堆垃圾值。。。
发生了意向不到的后果。所以本文来深入分析这个原理。
这段代码输出的是一个不确定的值,name与age的值并不是我们期待的,原因在于执行A("bob", 20)时,并不是用这一构造函数来初始化当前的内存区,而是初始化了一个临时对象的内存区。
具体阐述:A a;这里已经为a分配了内存,然后调用默认构造函数,但是默认构造函数还未执行完,却调用了另一个构造函数,这样相当于产生了一个匿名的临时A对象,它调用A("bob", 20)构造函数,将这个匿名临时对象自己的数据成员name与age初始化后,当执行完这一行,这个匿名临时对象已经被释放掉了,此时a的数据成员并没有得到初始化。于是a的name与age是未初始化的,因此其值也是不确定的。
那么如何在C++中实现构造函数调用构造函数呢?
两种方案,第一种:封装到公共操作到私有函数中;第二种:定点new;
我们将原来的代码改写:
class A {private: string name; int age; void init(string name, int age) { this->name = name; this->age = age; }public: A(string name, int age) { init(name, age); } A() { init("bob",20); } void print() { cout << name << " " << age << endl; } // ... // ...};此时输出就正常了,这种方案比较简单,相信搭建也明白是什么意思,就不多阐述了,重点阐述第二种方案!
new的另一种new的表达式,定点new表达式(placement new),它的作用是在已分配的原始内存中初始化一个对象,它与new的其他版本不同之处在于它并不分配内存。
表达式的原型如下:
new (place_address) type因此到这里可以使用如下:
new(this)A("bob", 20);最后的完整代码:
class A {private: string name; int age;public: A(string name, int age) { this->name = name; this->age = age; } A() { new(this)A("bob", 20); } void print() { cout << name << " " << age << endl; } // ... // ...};输出:
bob 20总结:针对构造互调两种方法解决:
定点new
封装到成员函数,推荐为private函数。