C++中初始化列表的使用(总结)
原文链接 https://www.cnblogs.com/dishengAndziyu/p/10906081.html
参考链接:https://www.cnblogs.com/laiqun/p/5776212.html
https://blog.csdn.net/zzyczzyc/article/details/87542418
https://www.cnblogs.com/cthon/p/9178701.html
http://blog.sina.com.cn/s/blog_4c79cc450100lkzh.html
1,初始化列表是在 C++ 中才引入的;
2,以“类中是否可以定义 const 成员?”这个问题来引入初始化列表:
1,const 这个关键字可以定义真正意义上的常量,也可以在某些情况下定义只读变量;
3,小实验:
1,下面的类定义是否合法?如果合法,ci 的值是什么,存储在哪里?


1 #include <stdio.h>
2
3 class Test
4 {
5 private:
6 const int ci; // const 作用于 C++ 中的成员变量后得到的是只读成员变量,只读成员变量是不可以出现在成员符号左边的;所以会出现第 10 行的错误信息;
7 public:
8 /*
9 Test() // 在这里编译器显示:ci 是一个 const 成员,没有进行初始化;因此如果要初始化 ci 成员变量,必须要在这一行进行,这个时候就让初始化列表出厂了;
10 {
11 ci = 10; // 在这里编译器显示:ci 在这个类中是一个只读的成员变量;
12 }
13 */
14
15 /* 由上面的语句改换如下 */
16 Test() : ci(10) // ci 在初始化之后可以改变,因为 ci 在这里只是一个只读的成员变量,仅仅是不能出现在赋值符号左边而已;我们依旧可以通过指针的方式来修改 ci 里面的值;
17 {
18 // ci = 10;
19 }
20
21 int getCI()
22 {
23 return ci;
24 }
25 };
26
27 int main()
28 {
29 Test t; // 当这里没有手工定义上面程序中的无参构造函数的时候,显示有“未初始化的 const 的成员”的错误; 同时的当这里没有通过类来定义对象的时候,可以通过编译,说明 const 可以修饰 C++ 中的成员变量;
30
31 printf("t.ci = %d\n", t.getCI());
32
33 return 0;
34 }


4,C++ 中提供了初始化列表对成员变量进行初始化,其语法规则为:
1,代码示例:
1 ClassName::ClassName() : m1(v1), m2(v1, v2), m3(v3) // 用 v1, (v1, v2), v3 分别对 m1, m2, m3 初始化;
2 {
3 // some other initialize operation;
4 }
1,初始化列表应该在构造函数的地方使用;
2,构造函数参数列表之后函数体之前定义初始化列表;
3,其作用就是对成员变量进行初始化;
2,注意事项(避免 bug 很重要):
1,成员的初始化顺序与成员的声明顺序相同;
2,成员的初始化顺序与初始化列表中的位置无关;
3,初始化列表先于构造函数的函数体执行;
(1)当构造函数的函数体开始执行的时候,对象已经创建完毕了,执行构造函数的函数体仅仅是为了初始化我们这个对象的状态而已;
(2)所以说初始化列表既然是用于初始化,那么必须在我们这个类对象创建的同时来进行执行,而不应该是对象已经创建好了才来进行一系列的初始化工作,这一点是有明确差异的,这个差异也就是初始化和赋值之间的差异;
5,初始化列表的使用编程实验:


1 #include <stdio.h>
2
3 class Value
4 {
5 private:
6 // int mi = 0; // 要初始化成员变量,只能使用初始化列表;在构造函 数当中的那是对成员变量赋值,不是初始化;
7
8 int mi;
9
10 public:
11 Value(int i)
12 {
13 printf("i = %d\n", i);
14 mi = i;
15 }
16
17 int getI()
18 {
19 return mi;
20 }
21 };
22
23 class Test
24 {
25 private:
26 /*
27 Value m2(2); // 这种明确的调用一个参数的方式也是有语法错误的;
28 Value m3(3);
29 Value m1(1);
30 */
31 Value m2;
32 Value m3;
33 Value m1;
34
35 public:
36 /*
37 Test() // 这里编译器显示没有 value 类的无参构造函数来匹配调用;
38 {
39
40 }
41 */
42 Test() : m1(1), m2(2), m3(3) // 成员变量的初始化必须通过初始化列表来完成;
43 {
44 printf("Test::Test()\n"); // 初始化列表先于构造函数的函数体执行;
45 }
46 };
47
48 int main()
49 {
50 Test t;
51
52 return 0;
53 }


6,类中的 const 成员:
1,类中的 const 成员会被分配空间的;在这里注意:(只有编译时 用立即数初始化的才是真的常量,const成员在编译时 没法直接赋值,能不能进符号表 就看编译时是否可以直接知道值,类只是一个模子 所以不可能给模子赋值 只有在创建对象时才可能 因为创建对象时是分配空间的时候)
(1)const 成员分配的空间和我们整个类对象分配的空间一致;
2,类中的 const 成员的本质是只读变量;
(1)根据编译提供的错误 bug 信息而得来;
3,类中的const 成员只能在初始化列表中指定初始值;
(1)编译器无法直接得到 const 成员的初始值,因此无法进入符号表成为真正意义上的常量;
(2) 运行时才来定义对象申请空间,调用构造函数,继而来调用初始化列表初始化成员变量;
7,只读成员变量编程实验:
1,代码示例:


1 #include <stdio.h>
2
3 class Value
4 {
5 private:
6 int mi;
7 public:
8 Value(int i)
9 {
10 printf("i = %d\n", i);
11 mi = i;
12 }
13 int getI()
14 {
15 return mi;
16 }
17 };
18
19 class Test
20 {
21 private:
22 const int ci;
23
24 Value m2;
25 Value m3;
26 Value m1;
27
28 public:
29 Test() : m1(1), m2(2), m3(3), ci(100)
30 {
31 printf("Test::Test()\n");
32 }
33
34 int getCI()
35 {
36 return ci;
37 }
38
39 int setCI(int v)
40 {
41 int* p = const_cast<int*>(&ci); // 通过指针来操作常量对象;
42
43 *p = v;
44 }
45 };
46
47
48 int main()
49 {
50 Test t;
51
52 printf("t.ci = %d\n", t.getCI());
53
54 t.setCI(10);
55
56 printf("t.ci = %d\n", t.getCI());
57
58 return 0;
59 }


这个实验说明:类中的 const 成员不是真正意义上的常量,它只是只读变量(编译器告诉的);
8。小插曲:
1,初始化与赋值不同:
1,初始化:对正在创建的对象进行初始值设置;初始化的时候,对象还没创建好,在创建的同时,我们将它的值确定了;
2,赋值:对已经存在的对象进行值设置;

1 int main()
2 {
3 int i = 0; // 这是初始化,初始化的时候 i 还不存在;
4 // ...
5 i = 0; // 这是赋值, i 是实际存在的了,并且 i 已经有一个值了,这个时候将 i 的值被改变了
6 }

9,小结:
1,类中可以使用初始化列表对成员进行初始化;
(1)类中不能直接初始化成员变量(不论变量为一般的还是类的对象),只能通过初始化列表来初始化;
2,初始化列表先于构造函数体执行;
3,类中可以定义 const 成员变量(这里是变量);
(1)const 作用于类的成员后得到的仅是只读变量;
4,const 成员变量必须在初始化列表中指定初始值;
5,const 成员变量为只读变量;
C++中初始化列表的使用(总结)的更多相关文章
- c++中初始化列表的初始化变量顺序问题
例题来看:请问下面程序打印出的结果是什么? #include <iostream> #include <string> using namespace std; class b ...
- C++中初始化列表的使用
1,初始化列表是在 C++ 中才引入的: 2,以“类中是否可以定义 const 成员?”这个问题来引入初始化列表: 1,const 这个关键字可以定义真正意义上的常量,也可以在某些情况下定义只读变量: ...
- C++类的成员初始化列表的相关问题
在以下四中情况下,要想让程序顺利编译,必须使用成员初始化列表(member initialization list): 1,初始化一个引用成员(reference member): 2,初始化一个常量 ...
- C++中使用初始化列表的情况
http://blog.csdn.net/iceshirley/article/details/5688696 要理解这个问题,从概念上,我们要知道一点,那就是构造函数的执行过程会分成两个阶段:隐式或 ...
- 【c++】必须在类初始化列表中初始化的几种情况
转自:http://www.cnblogs.com/kaituorensheng/p/3477630.html 1. 类成员为const类型 2. 类成员为引用类型 #include <iost ...
- C++-什么时候需要在类的构造函数中使用初始化列表
1,如果基类没有default构造函数,则意味着其不能自己初始化.如果其被派生,派生类的构造函数要负责调用基类的构造函数,并传递给它需要的参数.下例中Base 2,如果类成员没有默认构造函数.下例中E ...
- Effective C++学习笔记:初始化列表中成员列出的顺序和它们在类中声明的顺序相同
类成员的默认初始化顺序是按照声明顺序进行, 如果使用初始化列表初始化成员变量, 则必须按照成员变量的声明顺序进行; 否则, 在变量之间交替赋值时, 会产生, 未初始化的变量去赋值其他变量; 同时GCC ...
- C++中的初始化列表中可以对那些变量或对象进行初始化
构造函数与其函数体之间可以添加初始化列表,能对某些对象进行初始化.格式为 类名() : 变量1(参数1),变量2(参数2) { } 1. 父类的对象的构造必须在初始化列表中,如: 子类名(): ...
- C++中为什么构造函数初始化列表
已经有个构造函数负责初始化,为什么还需要构造函数初始化表呢? 在以下三种情况下需要使用初始化成员列表:一,需要初始化的数据成员是对象的情况:二,需要初始化const修饰的类成员:三,需要初始化引用成员 ...
随机推荐
- 什么是NFT?
我有一个年轻朋友,最近买了一个数字艺术品,9百多入手,几周后卖掉,赚了7万多,他告诉我这个东西叫NFT. 2021年twitter创始人杰克.多西将自己发布的第一条twitter通过NFT以250万美 ...
- 千万级 PV是什么意思?
首先介绍下pv的概念: PV(访问量):即Page View,页面刷新一次算一次. UV(独立访客):即Unique Visitor,00:00-24:00内相同的客户端只被计算一次. IP(独立IP ...
- 【高并发】两种异步模型与深度解析Future接口
大家好,我是冰河~~ 本文有点长,但是满满的干货,以实际案例的形式分析了两种异步模型,并从源码角度深度解析Future接口和FutureTask类,希望大家踏下心来,打开你的IDE,跟着文章看源码,相 ...
- python 逻辑运算及奇怪的返回值(not,and,or)
首先,'and'.'or'和'not'的优先级是not>and>or. 同一优先级从左往右计算. 先说非运算,Python的非运算与这些语言相比,并没有特别的地方.not只有两个返回值,T ...
- 『现学现忘』Docker基础 — 14、Docker的卸载
目录 1.查询Docker安装过的包 2.卸载Docker软件包 3.删除残留目录 4.验证是否卸载 5.20版本Docker卸载(官方文档) 1.查询Docker安装过的包 执行yum list i ...
- 基于 vite2 + Vue3 写一个在线帮助文档工具
提起帮助文档,想必大家都会想到 VuePress等,我也体验了一下,但是感觉和我的思路不太一样,我希望的是那种可以直接在线编辑文档,然后无需编译就可以直接发布的方式,另外可以在线写(修改)代码并且运行 ...
- Django-模板布局
- C++ STL vector扩容原理分析
扩容特点: 1)新增元素:vector通过一个连续的数组存放元素,如果集合已满,在新增数据的时候,就要分配一块更大的内存,将原来的数据复制过来,释放之前的内存,在插入新增的元素: 2)对vector的 ...
- BBS项目分布搭建三(个人站点时间归档补充,实现侧边栏跳转、无线级分类、实现文章详情页展示功能)
BBS项目分布搭建三(个人站点时间归档补充,) 1. 个人站点时间归档 """ settings.py设置最好更改以下: LANGUAGE_CODE = 'zh-hans ...
- [C++]C风格、C++风格和C++11特性的线程池
线程池概念 假设完成一项任务需要的时间=创建线程时间T1+线程执行任务时间T2+销毁线程时间T3,如果T1+T3的时间远大于T2,通常就可以考虑采取线程池来提高服务器的性能 thread pool就是 ...