继承构造函数
如果派生类并不需要改变构造函数,那么可以用新的using Base::Base
直接继承基类的构造函数,基类构造函数很多的时候,这样做非常方便。如果派生类还有自己的成员需要初始化,可以利用类成员初始化表达。
class A {
A(int a):a(a){}
int a;
};
class B :A
{
using A::A
}
使用继承构造函数,编译器不会产生真正的目标代码,可以节约目标代码控件。
继承构造函数参数的默认值不会被继承,而且会产生多个继承构造函数。A(int a = 1,double = 2)
的继承类使用using A::A
会产生B(int,double);B(int);B();B(const B &);
多个构造函数。因此,如果父类有默认参数的构造函数,使用继承构造需要很小心。
委派构造函数
在自己的构造函数里调用自己另外的构造函数,以便减少重复代码。
class A {
A () {init() ;}
A (int a ) :{init() ;this->a = a;}
A (char c ) :{init() ;this->c = c;}
};
使用tricky一点的方法是利用placement new
class A {
A(){init();}
A(int a):{ new (this )A();this ->a = a;}
A(char c):{
new (this )A();this ->c = c;}
};
class A {
A () {init() ;}
A (int a ) :A () { this->a = a;}
A (char c ) :A () {this->c = c;}
};
区别调用者和被调用者,A(int a)被称为 委派构造函数(delegate constructor),A()被称为 目标构造函数(target constructor)。
委派构造函数和初始化列表不能同时使用。比如A(int):A(),a(a){}
是非法的,初始化代码需要放到函数体。
委派构造函数互相调用会形成链状,但是不能成环,会导致编译错误。
移动语义和完美转发
移动语义:利用右值临时变量对变量进行拷贝构造时,变量偷走有时右值变量构造函数里创建的堆上的资源。右值临时变量是即将消失的变量,拿走堆上的资源不会产生问题,减少堆的拷贝。
class A {
A () { d = new Data ;}
A (const A & a ) :d(new Data (a .d ) ){}//普通拷贝
A (A && a ) :d(a .d ) { a.d = nullp