C++中类的构造函数调用顺序

2024-09-08 07:58

本文主要是介绍C++中类的构造函数调用顺序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

当建立一个对象时,首先调用基类的构造函数,然后调用下一个派生类的
构造函数,依次类推,直至到达派生类次数最多的派生次数最多的类的构造函数为止。
简而言之,对象是由“底层向上”开始构造的。因为,构造函数一开始构造时,总是
要调用它的基类的构造函数,然后才开始执行其构造函数体,调用直接基类构造函数时,
如果无专门说明,就调用直接基类的默认构造函数。在对象析构时,其顺序正好相反。

下面的这个程序说明这个问题


总结下来,我们必须明确的是当一个类继承与基类,并且自身还包含有其他类的成员对象的时候,构造函数的调用顺序为:调用基类的构造函数->调用成员对象的构造函数->调用自身的构造函数构造函数的调用次序完全不受构造函数初始化列表的表达式中的次序影响,与基类的声明次数和成员对象在函数中的声明次序有关


  1. C++子类和父类的构造函数调用顺序
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <iostream>  
  2. using namespace std;  
  3.   
  4. //子类  
  5. class base  
  6. {  
  7. public:  
  8.     base()  
  9.     {  
  10.         cout<<"i am base constuction!"<<endl;  
  11.     }  
  12. };  
  13. //父类  
  14. class drived : public base  
  15. {  
  16. public:  
  17.     drived()  
  18.     {  
  19.         cout<<"i am drived constuction!"<<endl;  
  20.     }  
  21. };  
  22. int main()  
  23. {  
  24.     drived d;  
  25.     //输出结果  
  26.     /* 
  27.     i am base constuction! 
  28.     i am drived constuction! 
  29.     Press any key to continue 
  30.     */  
  31.     return 0;  
  32. }  

由输出结果我们可以看到,先调用的是base类的构造函数,再调用了drived类的构造函数,也就是说,在声明子类实例的时候,是会先调用父类的构造函数的。这个很好理解,子类是包含了父类的信息的,所以要构造子类,必须先构造父类的信息,然后加入一些子类的新信息。

成员类的构造函数调用顺序
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <iostream>  
  2. using namespace std;  
  3.   
  4. class son  
  5.     {  
  6.     public:  
  7.         son()  
  8.         {  
  9.             cout<<"i am son constuction!"<<endl;  
  10.         }  
  11.     };  
  12. class family  
  13. {  
  14. public:  
  15.     family()  
  16.     {  
  17.         cout<<"i am family constuction!"<<endl;  
  18.     }  
  19.     son jack;  
  20.       
  21. };  
  22.   
  23.   
  24. int main()  
  25. {  
  26.     family happyfamily;  
  27.     //输出结果  
  28.     /* 
  29.     i am son constuction! 
  30.     i am family constuction! 
  31.     Press any key to continue 
  32.  
  33.     */  
  34.     return 0;  
  35. }  

从上面的结果我们科看到,先调用的是构造函数。这是为什么?因为类中含有成员类时,先要给成员类申请空间,先调用成员类的构造函数,然后再调用自身的构造函数。

[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <iostream>  
  2. using namespace std;  
  3. //父亲姓名  
  4. class name  
  5. {  
  6. public:  
  7.     name()  
  8.     {  
  9.         cout<<"i am name constuction!"<<endl;  
  10.     }  
  11. };  
  12. //子类  
  13. class base  
  14. {  
  15. public:  
  16.     base()  
  17.     {  
  18.         cout<<"i am base constuction!"<<endl;  
  19.     }  
  20. };  
  21. //父类  
  22. class drived : public base  
  23. {  
  24. public:  
  25.     drived()  
  26.     {  
  27.         cout<<"i am drived constuction!"<<endl;  
  28.     }  
  29.     name drivedname;  
  30. };  
  31.   
  32. int main()  
  33. {  
  34.     drived d;  
  35.     //输出结果  
  36.     /* 
  37.     i am base constuction! 
  38.     i am name constuction! 
  39.     i am drived constuction! 
  40.     Press any key to continue 
  41.  
  42.     */  
  43.     return 0;  
  44. }  

上面的例子是声明了一个name类,一个base类,一个drived类,其中drived类是从base类派生,且drived类中声明了name类的一个实例。
那么根据我们上面的分析,我们声明了drived类之后,先会调用父类构造函数,也就是输出i am base construction,然后调用自身的成员类构造函数,也就是i am name construction,最后调用自身的构造函数,也就是i am drived construction。
同理,如果我们在父类base类中声明了name成员类的话,顺序会是name->base->drived。我下面给出测试例子:
[cpp]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <iostream>  
  2. using namespace std;  
  3. //父亲姓名  
  4. class name  
  5. {  
  6. public:  
  7.     name()  
  8.     {  
  9.         cout<<"i am name constuction!"<<endl;  
  10.     }  
  11. };  
  12. //子类  
  13. class base  
  14. {  
  15. public:  
  16.     base()  
  17.     {  
  18.         cout<<"i am base constuction!"<<endl;  
  19.     }  
  20.     name drivedname;  
  21. };  
  22. //父类  
  23. class drived : public base  
  24. {  
  25. public:  
  26.     drived()  
  27.     {  
  28.         cout<<"i am drived constuction!"<<endl;  
  29.     }  
  30.       
  31. };  
  32.   
  33. int main()  
  34. {  
  35.     drived d;  
  36.     //输出结果  
  37.     /* 
  38.     i am name constuction! 
  39.     i am base constuction! 
  40.     i am drived constuction! 
  41.     Press any key to continue 
  42.     */  
  43.     return 0;  
  44. }  


这篇关于C++中类的构造函数调用顺序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1147552

相关文章

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Spring Bean初始化及@PostConstruc执行顺序示例详解

《SpringBean初始化及@PostConstruc执行顺序示例详解》本文给大家介绍SpringBean初始化及@PostConstruc执行顺序,本文通过实例代码给大家介绍的非常详细,对大家的... 目录1. Bean初始化执行顺序2. 成员变量初始化顺序2.1 普通Java类(非Spring环境)(

C++中detach的作用、使用场景及注意事项

《C++中detach的作用、使用场景及注意事项》关于C++中的detach,它主要涉及多线程编程中的线程管理,理解detach的作用、使用场景以及注意事项,对于写出高效、安全的多线程程序至关重要,下... 目录一、什么是join()?它的作用是什么?类比一下:二、join()的作用总结三、join()怎么

C++中全局变量和局部变量的区别

《C++中全局变量和局部变量的区别》本文主要介绍了C++中全局变量和局部变量的区别,全局变量和局部变量在作用域和生命周期上有显著的区别,下面就来介绍一下,感兴趣的可以了解一下... 目录一、全局变量定义生命周期存储位置代码示例输出二、局部变量定义生命周期存储位置代码示例输出三、全局变量和局部变量的区别作用域

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

浅析Spring如何控制Bean的加载顺序

《浅析Spring如何控制Bean的加载顺序》在大多数情况下,我们不需要手动控制Bean的加载顺序,因为Spring的IoC容器足够智能,但在某些特殊场景下,这种隐式的依赖关系可能不存在,下面我们就来... 目录核心原则:依赖驱动加载手动控制 Bean 加载顺序的方法方法 1:使用@DependsOn(最直

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window