前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >C++11继承构造函数

C++11继承构造函数

作者头像
恋喵大鲤鱼
发布2018-08-16 16:02:56
1.8K0
发布2018-08-16 16:02:56
举报
文章被收录于专栏:C/C++基础C/C++基础

1.简介

子类为完成基类初始化,在C++11之前,需要在初始化列表调用基类的构造函数,从而完成构造函数的传递。如果基类拥有多个构造函数,那么子类也需要实现多个与基类构造函数对应的构造函数。

代码语言:javascript
复制
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       
    }
};

书写多个派生类构造函数只为传递参数完成基类的初始化,这种方式无疑给开发人员带来麻烦,降低了编码效率。从C++11开始,推出了继承构造函数(Inheriting Constructor),使用using来声明继承基类的构造函数,我们可以这样书写。

代码语言:javascript
复制
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把基类构造函数继承到派生类中,不再需要书写多个派生类构造函数来完成基类的初始化。更为巧妙的是,C++11标准规定,继承构造函数与类的一些默认函数(默认构造、析构、拷贝构造函数等)一样,是隐式声明,如果一个继承构造函数不被相关代码使用,编译器不会为其产生真正的函数代码。这样比通过派生类构造函数“透传构造函数参数”来完成基类初始化的方案,总是需要定义派生类的各种构造函数更加节省目标代码空间。

2.注意事项

(1)继承构造函数无法初始化派生类数据成员。 继承构造函数的功能是初始化基类,对于派生类数据成员的初始化则无能为力。解决的办法主要有两个: 一是使用C++11特性就地初始化成员变量,可以通过=、{}对非静态成员快速地就地初始化,以减少多个构造函数重复初始化变量的工作,注意初始化列表会覆盖就地初始化操作。

代码语言:javascript
复制
class Derived :public Base
{
public:
    //使用继承构造函数
    using Base::Base;

    //假设派生类只是添加了一个普通的函数
    void display()
    {
//dosomething       
    }
private:
//派生类新增数据成员
double m_double{0.0};
};

二是新增派生类构造函数,使用构造函数初始化列表初始化。

代码语言:javascript
复制
class Derived :public Base
{
public:
    //使用继承构造函数
    using Base::Base;

//新增派生类构造函数
Derived(int a,double b):Base(a),m_double(b){}

    //假设派生类只是添加了一个普通的函数
    void display()
    {
//dosomething       
    }
private:
//派生类新增数据成员
double m_double{0.0};

相比之下,第二种方法需要新增构造函数,明显没有第一种方法简洁,但第二种方法可由用户控制初始化值,更加灵活。各有优劣,两种方法需结合具体场景使用。

(2)构造函数拥有默认值会产生多个构造函数版本,且继承构造函数无法继承基类构造函数的默认参数,所以我们在使用有默认参数构造函数的基类时就必须要小心。

代码语言:javascript
复制
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中的构造函数会有下面几个版本:

代码语言:javascript
复制
A()
A(int)
A(int,double)
A(constA&)

那么B中对应的继承构造函数将会包含如下几个版本:

代码语言:javascript
复制
B()
B(int)
B(int,double)
B(constB&)

可以看出,参数默认值会导致多个构造函数版本的产生,因此在使用时需格外小心。

(3)多继承的情况下,继承构造函数会出现“冲突”的情况,因为多个基类中的部分构造函数可能导致派生类中的继承构造函数的函数名、参数(即函数签名)相同。考察如下代码:

代码语言:javascript
复制
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)
    C(int i):A(i),B(i){}
};

为避免继承构造函数冲突,可以通过显示定义继承类冲突的构造函数,组织隐式生成相应的继承构造函数。

此外,使用继承构造函数时,还需要注意以下几点: (1)如果基类构造函数被申明为私有成员函数,或者派生类是从基类中虚继承的 ,那么就不能在派生类中申明继承构造函数; (2)一旦使用继承构造函数,编译器就不会再为派生类生成默认构造函数了。

参考文献

[1]深入理解C++11

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018年08月15日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 1.简介
  • 2.注意事项
  • 参考文献
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档