C++中对象的构造顺序
1,C++ 中的类可以定义多个对象,那么对象构造顺序是怎样的?
1,很多的 bug 是由对象的构造顺序造成的,虽然它不难;
2,对象的构造往往和构造函数牵涉在一起,构造函数的函数体又可能由非常复杂的程序逻辑组成的;
3,这样就有可能引入了一个问题,不同类的它们的构造函数中的程序逻辑也许是相互依赖的,当这种相互依赖一旦发生,那么对象的构造顺序就很可能导致程序中非常难以调试的 bug 出现;
4,在工程中,由于对象的构造顺序而造成的软件 bug 非常之多,因此有必要梳理下对象的构造顺序;
2,对于局部对象:
1,当程序执行流到达对象的定义语句时进行构造;
1,构造时调用构造函数;
3,局部对象的构造顺序实例分析:
1,代码示例:
#include <stdio.h> class Test
{
private:
int mi;
public:
Test(int i)
{
mi = i;
printf("Test(int i): %d\n", mi); // 打印语句对实验非常有用;
} Test(const Test& obj)
{
mi = obj.mi;
printf("Test(const Test& obj): %d\n", mi);
}
}; int main()
{
int i = ;
Test a1 = i; // 第一个被构造的对象 while( i < )
{
Test a2 = ++i; // 第二个被构造的对象;反复的被构造三次对象;
} // goto End; // 下面的输出不会执行了;程序执行流跳过了程序调用的定义,则对象就没有被构造了;
if( i < )
{
Test a = a1; // 第三个被构造的对象;Test(const Test& obj): 3
}
else
{
Test a();
}
// End:
return ;
}
2,这个实验说明:
1,程序执行流直接和局部对象的构造顺序息息相关,如果非法的改变程序执行流,那么程序可能产生灾难性的错误,参见如下示例:
#include <stdio.h> class Test
{
private:
int mi;
public:
Test(int i)
{
mi = i;
printf("Test(int i): %d\n", mi);
} Test(const Test& obj)
{
mi = obj.mi;
printf("Test(const Test& obj): %d\n", mi);
} int getMi()
{
return mi;
}
}; int main()
{
int i = ;
Test a1 = i; // Test(int i): 0 while( i < )
{
Test a2 = ++i; // Test(int i): 1, 2, 3
}
goto End;
Test a(); // 这里程序执行流跳过了对象的定义,因此 a 这个对象是没有被构造的,没有被构造意味着 a 这个对象它的状态是没有被初始化的,如果后面使用这个对象,则会产生灾难性的错误;g++ 可以帮助我们报出这个错误,但是其它编译器有可能不能,因为这不是标准,比如 Visual Studio 中的编译器;
End:
printf("a.mi = %d\n", a.getMi()); // 这里打印随机值;a 对象没有被初始化,因为构造函数没有被调用; return ;
}
2,这个实验说明:
1,对象的构造如果不是显示的调用,则只发生在对象定义(运行时)之时;
2,编译时只产生没有初始化的对象(这里很模糊,不清楚);
3,程序执行流和程序的编译不相关,编译还是会都编译的。执行确定了构造函数的调用,而编译则是生成空间;
4,对于堆对象:
1,当程序执行流到达 new 语句时创建对象;
1,创建对象就要触发构造函数的调用;
2,其实堆对象构造顺序也和程序执行流相关,只不过说由于引入了 new 关键字,堆对象的构造顺序比局部对象更容易确认些;
2,使用 new 创建对象将自动触发构造函数的调用;
5,堆对象的构造顺序编程实验:
#include <stdio.h> class Test
{
private:
int mi;
public:
Test(int i)
{
mi = i;
printf("Test(int i): %d\n", mi);
} Test(const Test& obj)
{
mi = obj.mi;
printf("Test(const Test& obj): %d\n", mi);
} int getMi()
{
return mi;
}
}; int main()
{
int i = ;
Test* a1 = new Test(i); // Test(int i): 0 while( ++i < )
if( i % )
new Test(i); // Test(int i): 1, 3, 5, 7, 9 if( i < )
new Test(*a1);
else
new Test(); // Test(int i): 100 return ;
}
1,堆对象的创建也会受到 goto 语句的影响,因为 goto 语句会改变程序执行流,也就可能绕过堆对象的创建;
6,对于全局对象:
1,全局对象的构造顺序是不确定的;
1,这里带来的 bug 是非常多的;
2,C++ 标准没有定义全局对象的构造顺序;
2,不同的编译器使用不同的规则确定构造顺序;
7,全局对象的构造顺序:
1,test.h文件:
#ifndef _TEST_H_
#define _TEST_H_ #include <stdio.h> class Test
{
public:
Test(const char* s)
{
printf("%s\n", s);
}
}; #endif
2,t1.cpp 文件:
#include "test.h"
Test t1("t1");
3,t2.cpp 文件:
#include "test.h"
Test t2("t2");
4,t3.cpp 文件:
#include "test.h"
Test t3("t3");
5,主函数:
#include "test.h"
Test t4("t4"); // 全局对象的构造一定会在 main 函数之前完成,和程序执行流没有关系;main() 函数执行之前,没有程序执行流的概念,如果出现多个全局对象的时候,它们的构造顺序就不确定了;尤其是两个不同的操作系统更是不同;
int main()
{
Test t5("t5");
}
1,在以后的开发过程中,要避免全局对象之间的相互依赖;
2,当今的软件开发领域,尽量的不使用全局变量的,其中原因之一就是全局的变量它们的初始化顺序是不确定的,面向对象领域就变成了全局对象的构造顺序是不确定的;
3,goto 语句也是禁用的,因为它会改变程序执行流,这样会造成局部对象构造顺序产生错乱,有可能导致程序里面局部对象构造出现问题,进而产生不可预计的灾难性错误;
8,小结:
1,局部对象的构造顺序依赖于程序的执行流;
2,堆对象的构造顺序依赖于 new 的使用顺序;
3,全局对象的构造顺序是不确定的;
C++中对象的构造顺序的更多相关文章
- C++类继承中,基类/当前对象属性/当前对象的构造顺序
[1]中提到,规范的派生类构造函数三个要点: 首先创建基类对象 应通过成员初始化列表,创建基类对象 应该初始化本派生类新增的成员变量 那在构造派生类实例的过程中,其基类(以及多继承的时候多个基类)/当 ...
- js中对象的输出顺序
前言:最近用for-in时,看到说for-in不能保证遍历的对象顺序,对此有些疑问,于是便研究了下,本文做简要说明. 现象 let obj = { a: 'a', b: 'b', 1: 1, 2: 2 ...
- c++中对象的构造和销毁
对象的初始化 如下 ckasss Person { public: ]; char sex; int age; }; Person p={}; //对象初始化 构造数组对象时,需要一个没有参数的构造函 ...
- 10.C++-构造函数初始化列表、类const成员、对象构造顺序、析构函数
首先回忆下,以前学的const 单独使用const修饰变量时,是定义的常量,比如:const int i=1; 使用volatile const修饰变量时,定义的是只读变量 使用const & ...
- C++解析(12):初始化列表与对象构造顺序、析构顺序
0.目录 1.类成员的初始化 2.类中的const成员 3.对象的构造顺序 3.1 局部对象的构造顺序 3.2 堆对象的构造顺序 3.3 全局对象的构造顺序 4.对象的析构顺序 5.小结 1.类成员的 ...
- C++ 对象构造顺序、构析函数、临时对象。
对象的构造顺序: 1.对于局部对象,构造顺序是根据程序执行流进行构造,从上到下. #include <stdio.h> class Test { int mi; public: Test( ...
- [百度空间] [转] 在 Visual C++ 中控制全局对象的初始化顺序
from: http://blog.csdn.net/classfactory/archive/2004/08/07/68202.aspx 在 C++ 中,同一个翻译单位(.cpp文件)里的全局对象的 ...
- C++深度解析教程学习笔记(6)对象的构造和销毁
1. 对象的初始化 (1)从程序设计的角度看,对象只是变量,因此: ①在栈上创建对象时,成员变量初始化为随机值 ②在堆上创建对象时,成员变量初始化为随机值 ③在静态存储区创建对象时,成员变量初始化为 ...
- Java中对象构造
构造函数 作用:在构造对象的同时初始化对象.java强制要求对象 诞生同时被初始化,保证数据安全. 调用过程和机制:①申请内存,②执行构造函数的函数体,③返回对象的引用. 特点:与类同名,无返回类型, ...
随机推荐
- JAVA笔记28-正则表达式(补充、不重要)
一.Greedy(贪婪的)尽可能多的匹配,Reluctant(不情愿的)尽可能少的匹配.Possessive(独占的)不常用. Greedy 数量词 X? X,一次或一次也没有 X* X,零次或多次 ...
- 【NOIP2017模拟6.25】小W的动漫
题目 小W最近迷上了日本动漫,每天都有无数部动漫的更新等着他去看,所以他必须将所有的动漫排个顺序,当然,虽然有无数部动漫,但除了1号动漫,每部动漫都有且仅有一部动漫是它的前传(父亲),也就是说,所有的 ...
- Python 元组Ⅱ
删除元组 元组中的元素值是不允许删除的,但我们可以使用del语句来删除整个元组,如下实例: 以上实例元组被删除后,输出变量会有异常信息,输出如下所示: 元组运算符 与字符串一样,元组之间可以使用 + ...
- python 习题
文件内容为一个多层元组,遍历该元组,当全为数字时输出数字之和,全为字母输出字符串,有数字有字母输出False,并将该内容写入到该文件的下一行中 # 方法一: t1= ((1,2,3),("a ...
- Mysql 连接路径 url 参数解析
1.mysql - url 参数解析 url:jdbc:mysql://127.0.0.1:3306/user?useUnicode=true&characterEncoding=utf8 u ...
- 兄弟连教育分享-SQL性能优化十条经验
1.查询的模糊匹配 尽量避免在一个复杂查询里面使用 LIKE '%parm1%'——红色标识位置的百分号会导致相关列的索引无法使用,最好不要用. 兄弟连教育分享-SQL性能优化十条经验 解决办法: 其 ...
- PHP入门培训教程 PHP变量及常量
一.PHP5.4的基本语法格式 1.PHP的分割符 $php=true; //分号结束语句 if($php){ echo "真"; //分号结束语句 } //大括号结束语 ...
- HDU 6651 Final Exam
hdu题面 Time limit 2000 ms Memory limit 524288 kB OS Windows 吐槽 比赛时候晕死了-- 解题思路 先留坑 公式法 https://blog.cs ...
- Java网络编程之Netty服务端ChannelOption.SO_BACKLOG配置
ChannelOption.SO_BACKLOG对应的是tcp/ip协议listen函数中的backlog参数,函数listen(int socketfd,int backlog)用来初始化服务端可连 ...
- k8s中pod内dns无法解析的问题
用k8s创建了pod,然后进入pod后,发现在pod中无法解析www.baidu.com,也就是出现了无法解析外面的域名的问题.经过高人指点,做个小总结.操作如下. 一,将CoreDNS 的Confi ...