news 2026/1/1 12:37:01

C++11新特性(十二)——委托构造函数和继承构造函数

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C++11新特性(十二)——委托构造函数和继承构造函数

目录

1 委托构造函数

2 继承构造函数


1 委托构造函数

委托构造函数允许使用同一个类中的一个构造函数调用其他的构造函数,从而简化相关变量的初始化。下面通过一个例子说明:

class Test { public: Test() {}; Test(int max) { this->m_max = max > 0 ? max : 100; } Test(int max, int min) { this->m_max = max > 0 ? max : 100; // 冗余代码 this->m_min = min > 0 && min < max ? min : 1; } Test(int max, int min, int mid) { this->m_max = max > 0 ? max : 100; // 冗余代码 this->m_min = min > 0 && min < max ? min : 1; // 冗余代码 this->m_mid = mid < max && mid > min ? mid : 50; } public: int m_max; int m_min; int m_mid; }; int main() { Test t(90, 30, 60); cout << "min: " << t.m_min << ", middle: " << t.m_mid << ", max: " << t.m_max << endl; return 0; }

在上面三个重载的构造函数中都出现重复的代码,在 C++11 以前,我们会把重复的代码放到一个函数里,然后调用;在 C++11 以后,加入了委托构造,我们就可以轻松地完成代码的优化了

class Test { public: Test() {}; Test(int max) { this->m_max = max > 0 ? max : 100; } Test(int max, int min) : Test(max) { this->m_min = min > 0 && min < max ? min : 1; } Test(int max, int min, int mid) : Test(max,min) { this->m_mid = mid < max && mid > min ? mid : 50; } public: int m_max; int m_min; int m_mid; }; int main() { Test t(90, 30, 60); cout << "min: " << t.m_min << ", middle: " << t.m_mid << ", max: " << t.m_max << endl; return 0; }

修改完代码之后,冗余的代码全部都没有了,并且在一个构造函数中调用了其他的构造函数用于相关数据的初始化,相当于是一个链式调用。在使用委托构造函数的时候还需要注意一些几个问题

  • 这种链式的构造函数调用不能形成一个闭环,否则会在运行期间抛出异常
  • 如果要进行多层构造函数的链式调用,建议将构造函数的调用的写在初始列表中而不是函数体内部,否则编译器会提示形参的重复定义。
Test() {}; Test(int max) { this->m_max = max > 0 ? max : 100; } Test(int max, int min) { Test(max); //报错 this->m_min = min > 0 && min < max ? min : 1; } Test(int max, int min, int mid) { Test(max, min); //报错 this->m_mid = mid < max && mid > min ? mid : 50; }
  • 在初始化列表中调用了代理构造函数初始化某个类成员变量之后,就不能在初始化列表中再次初始化这个变量了
// 错误, 使用了委托构造函数就不能再次m_max初始化了 Test(int max, int min) : Test(max), m_max(max) { this->m_min = min > 0 && min < max ? min : 1; }

2 继承构造函数

C++11 中提供的继承构造函数可以让派生类直接使用基类的构造函数,而无需自己再写构造函数尤其是在基类有很多构造函数的情况下,可以极大地简化派生类构造函数的编写。先来看没有继承构造函数之前的处理方式

class Base { public: Base(int i) :m_i(i) {} Base(int i, double j) :m_i(i), m_j(j) {} Base(int i, double j, string k) :m_i(i), m_j(j), m_k(k) {} int m_i; double m_j; string m_k; }; class Child : public Base { public: Child(int i) :Base(i) {} Child(int i, double j) :Base(i, j) {} Child(int i, double j, string k) :Base(i, j, k) {} }; int main() { Child c(520, 13.14, "i love you"); cout << "int: " << c.m_i << ", double: " << c.m_j << ", string: " << c.m_k << endl; return 0; }

我们通过继承构造函数在子类中重新定义和基类一样的构造函数:通过使用using 基类名::基类的构造函数名来使用基类的构造函数,这样子类中就可以不定义相同的构造函数了,直接使用基类的构造函数来构造派生类对象

class Base { public: Base(int i) :m_i(i) {} Base(int i, double j) :m_i(i), m_j(j) {} Base(int i, double j, string k) :m_i(i), m_j(j), m_k(k) {} int m_i; double m_j; string m_k; }; class Child : public Base { public: using Base::Base; // 引入父类Base的构造函数及其重载 }; int main() { Child c(520, 13.14, "i love you"); cout << "int: " << c.m_i << ", double: " << c.m_j << ", string: " << c.m_k << endl; return 0; }

在修改后的子类中,没有添加任何构造函数,而是添加了using Base::Base这样就可以在子类中直接继承父类的所有的构造函数,通过他们去构造子类对象了。

此外,如果在子类中隐藏了父类中的同名函数,也可以通过 using 的方式在子类中使用基类中的这些父类函数

class Base { public: Base(int i) :m_i(i) {} Base(int i, double j) :m_i(i), m_j(j) {} Base(int i, double j, string k) :m_i(i), m_j(j), m_k(k) {} void func(int i) { cout << "base class: i = " << i << endl; } void func(int i, string str) { cout << "base class: i = " << i << ", str = " << str << endl; } int m_i; double m_j; string m_k; }; class Child : public Base { public: using Base::Base; // 引入父类Base的构造函数及其重载 using Base::func; // 引入父类Base的func函数及其重载 void func() { cout << "child class: i'am huang" << endl; } }; int main() { Child c(250); c.func(); c.func(18); c.func(18, "huang"); return 0; }

输出结果

child class: i'am huang base class: i = 18 base class: i = 18, str = huang

子类中的func()函数隐藏了基类中的两个func(),默认情况下通过子类对象只能调用无参的func(),在上面的子类代码中添加了using Base::func之后,就可以通过子类对象直接调用父类中被隐藏的带参 func() 函数了。

这里插入一个注意事项,在 C++ 中,当我们在派生类中没有重写基类的重载函数之一的时候,在派生类中调用重载函数时是可以在其基类中查到的。然而,当我们想在子类中重写基类中某一个重载函数时,基类所有同名的重载函数会被隐藏,需通过using 基类名::函数名引入,才能让子类对象正常调用基类的重载函数。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2025/12/16 11:04:47

如何高效配置Flutter CanvasKit渲染方案:5个关键优化策略

如何高效配置Flutter CanvasKit渲染方案&#xff1a;5个关键优化策略 【免费下载链接】engine The Flutter engine 项目地址: https://gitcode.com/gh_mirrors/eng/engine 还在为Flutter Web应用在复杂场景下的渲染卡顿和内存占用过高而困扰吗&#xff1f;作为基于WebAs…

作者头像 李华
网站建设 2025/12/16 11:04:37

光纤耦合装置的公差分析

摘要在现代光学中&#xff0c;光纤存在于各种光学系统中&#xff0c;能够将多少光耦合到光纤中一直是人们关注的问题。耦合效率对系统的对准十分敏感&#xff0c;特别是对于芯径相对较小的单模光纤。在本例中&#xff0c;我们选择了一个设计良好的光纤耦合透镜&#xff0c;并根…

作者头像 李华
网站建设 2025/12/16 11:04:05

Dify可视化AI应用开发平台的核心优势全面揭秘

Dify可视化AI应用开发平台的核心优势全面揭秘 在大模型技术席卷全球的今天&#xff0c;企业对AI落地的期待从未如此迫切。然而现实却常常令人沮丧&#xff1a;一个看似简单的智能客服系统&#xff0c;动辄需要数周甚至数月的开发周期&#xff1b;提示词稍作调整&#xff0c;整个…

作者头像 李华
网站建设 2025/12/16 11:04:03

3个结构化数据技巧,让你的GEO收录率飙升200%

在AI主导的信息获取时代&#xff0c;GEO&#xff08;生成引擎优化&#xff09;已成为企业内容战略的核心战场。据水滴互动2025年行业白皮书数据显示&#xff0c;采用结构化数据优化的企业&#xff0c;其内容被AI引擎引用的概率提升2.3倍&#xff0c;决策链路转化率提高47%。本文…

作者头像 李华
网站建设 2025/12/16 11:03:45

Linux实时调度:3大策略对比与实战配置指南 [特殊字符]

Linux实时调度&#xff1a;3大策略对比与实战配置指南 &#x1f680; 【免费下载链接】linux-insides-zh Linux 内核揭秘 项目地址: https://gitcode.com/gh_mirrors/li/linux-insides-zh 还在为系统响应延迟而烦恼&#xff1f;Linux内核的实时调度机制正是你需要的解决…

作者头像 李华