长沙外贸企业网站建设,wordpress音乐播放器插件,wordpress 5.0网易云音乐,搜索引擎营销原理是什么继承的构造函数
1. 简介#xff1a;
子类为完成基类初始化#xff0c;在C11之前#xff0c;需要在初始化列表调用基类的构造函数#xff0c;从而完成构造函数的传递。如果基类拥有多个构造函数#xff0c;那么子类也需要实现多个与基类构造函数对应的构造函数。
class …继承的构造函数
1. 简介
子类为完成基类初始化在C11之前需要在初始化列表调用基类的构造函数从而完成构造函数的传递。如果基类拥有多个构造函数那么子类也需要实现多个与基类构造函数对应的构造函数。
class Base
{
public:Base(int va) : m_value(va), m_c(‘0’) {}Base(char c) : m_c(c), m_value(0) {}
private:int m_value;char m_c;
};class Derived : public Base
{
public:// 初始化基类需要透传基类的各个构造函数那么这是很麻烦的Derived(int va) : Base(va) {}Derived(char c) : Base(c) {}void display() { /* dosomething */ } //假设派生类只是添加了一个普通的函数
};书写多个派生类构造函数只为传递参数完成基类的初始化这种方式无疑给开发人员带来麻烦降低了编码效率。从C11开始推出了继承构造函数Inheriting Constructor使用using来声明继承基类的构造函数我们可以这样书写。
class Base {
public:Base(int va) : m_value(va), m_c(0) {}Base(char c) : m_c(c), m_value(0) {}
private:int m_value;char m_c;
};class Derived : public Base {
public: using Base::Base; //使用继承构造函数 void display() { /* dosomething */ } //假设派生类只是添加了一个普通的函数
};上面代码中我们通过using Base::Base把基类构造函数继承到派生类中不再需要书写多个派生类构造函数来完成基类的初始化。更为巧妙的是C11标准规定继承构造函数与类的一些默认函数默认构造、析构、拷贝构造函数等一样是隐式声明如果一个继承构造函数不被相关代码使用编译器不会为其产生真正的函数代码。这样比通过派生类构造函数“透传构造函数参数”来完成基类初始化的方案总是需要定义派生类的各种构造函数更加节省目标代码空间。 2. 注意事项
1继承构造函数无法初始化派生类数据成员。
继承构造函数的功能是初始化基类对于派生类数据成员的初始化则无能为力。解决的办法主要有两个一是使用C11特性就地初始化成员变量可以通过、{}对非静态成员快速地就地初始化以减少多个构造函数重复初始化变量的工作注意初始化列表会覆盖就地初始化操作。
class Derived :public Base
{
public:using Base::Base; //使用继承构造函数 void display() { /* dosomethin */ } //假设派生类只是添加了一个普通的函数
private:double m_double{ 0.0 }; //派生类新增数据成员
};2构造函数拥有默认值会产生多个构造函数版本且继承构造函数无法继承基类构造函数的默认参数所以我们在使用有默认参数构造函数的基类时就必须要小心。
class A
{
public:A(int a 3, double b 4) : m_a(a), m_b(b) {}void display() { cout m_a m_b endl; }private:int m_a;double m_b;
};class B : public A
{
public:using A::A;
};
那么A中的构造函数会有下面几个版本 1 2 3 4 A() A(int) A(int,double) A(constA)
那么B中对应的继承构造函数将会包含如下几个版本 1 2 3 4 B() B(int) B(int,double) B(constB)
可以看出参数默认值会导致多个构造函数版本的产生因此在使用时需格外小心。 3多继承的情况下继承构造函数会出现“冲突”的情况因为多个基类中的部分构造函数可能导致派生类中的继承构造函数的函数名、参数即函数签名相同。考察如下代码
class A {
public:A(int i) {}
};class B {
public:B(int i) {}
};class C : public A, public B {
public:using A::A;using B::B; //编译出错重复定义C(int)C(int i) :A(i), B(i) {} //显示定义继承构造函数C(int)
};
为避免继承构造函数冲突可以通过显示定义继承类冲突的构造函数组织隐式生成相应的继承构造函数。
此外使用继承构造函数时还需要注意以下几点
如果基类构造函数被申明为私有成员函数或者派生类是从基类中虚继承的 那么就不能在派生类中申明继承构造函数一旦使用继承构造函数编译器就不会再为派生类生成默认构造函数了。3. 测试代码
#include iostream
using namespace std;class Base {
public:Base(int va) : m_value(va), m_c(0) {}Base(char c) : m_c(c), m_value(0) {}int m_value;char m_c;
};class Derived : public Base {
public:using Base::Base;void display() { cout m_value endl m_c endl; }
};int main()
{Derived d(5);d.display();return 0;
}
输出结果