继承构造函数
- 如果派生类并不需要改变构造函数,那么可以用新的
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)