1,初始化列表是在 C++ 中才引入的;

2,以“类中是否可以定义 const 成员?”这个问题来引入初始化列表:

1,const 这个关键字可以定义真正意义上的常量,也可以在某些情况下定义只读变量;

3,小实验:

1,下面的类定义是否合法?如果合法,ci 的值是什么,存储在哪里?

 #include <stdio.h>

 class Test
{
private:
const int ci; // const 作用于 C++ 中的成员变量后得到的是只读成员变量,只读成员变量是不可以出现在成员符号左边的;所以会出现第 10 行的错误信息;
public:
/*
Test() // 在这里编译器显示:ci 是一个 const 成员,没有进行初始化;因此如果要初始化 ci 成员变量,必须要在这一行进行,这个时候就让初始化列表出厂了;
{
ci = 10; // 在这里编译器显示:ci 在这个类中是一个只读的成员变量;
}
*/ /* 由上面的语句改换如下 */
Test() : ci() // ci 在初始化之后可以改变,因为 ci 在这里只是一个只读的成员变量,仅仅是不能出现在赋值符号左边而已;我们依旧可以通过指针的方式来修改 ci 里面的值;
{
// ci = 10;
} int getCI()
{
return ci;
}
}; int main()
{
Test t; // 当这里没有手工定义上面程序中的无参构造函数的时候,显示有“未初始化的 const 的成员”的错误; 同时的当这里没有通过类来定义对象的时候,可以通过编译,说明 const 可以修饰 C++ 中的成员变量; printf("t.ci = %d\n", t.getCI()); return ;
}

4,C++ 中提供了初始化列表对成员变量进行初始化,其语法规则为:

1,代码示例:

 ClassName::ClassName() : m1(v1), m2(v1, v2), m3(v3)  // 用 v1, (v1, v2), v3 分别对 m1, m2, m3 初始化;
{
// some other initialize operation;
}

1,初始化列表应该在构造函数的地方使用;

2,构造函数参数列表之后函数体之前定义初始化列表;

3,其作用就是对成员变量进行初始化;

2,注意事项(避免 bug 很重要):

1,成员的初始化顺序与成员的声明顺序相同;

2,成员的初始化顺序与初始化列表中的位置无关;

3,初始化列表先于构造函数的函数体执行;

1,当构造函数的函数体开始执行的时候,对象已经创建完毕了,执行构造函数的函数体仅仅是为了初始化我们这个对象的状态而已;

2,所以说初始化列表既然是用于初始化,那么必须在我们这个类对象创建的同时来进行执行,而不应该是对象已经创建好了才来进行一系列的初始化工作,这一点是有明确差异的,这个差异也就是初始化和赋值之间的差异;

5,初始化列表的使用编程实验:

 #include <stdio.h>

 class Value
{
private:
// int mi = 0; // 要初始化成员变量,只能使用初始化列表;在构造函 数当中的那是对成员变量赋值,不是初始化; int mi; public:
Value(int i)
{
printf("i = %d\n", i);
mi = i;
} int getI()
{
return mi;
}
}; class Test
{
private:
/*
Value m2(2); // 这种明确的调用一个参数的方式也是有语法错误的;
Value m3(3);
Value m1(1);
*/
Value m2;
Value m3;
Value m1; public:
/*
Test() // 这里编译器显示没有 value 类的无参构造函数来匹配调用;
{ }
*/
Test() : m1(), m2(), m3() // 成员变量的初始化必须通过初始化列表来完成;
{
printf("Test::Test()\n"); // 初始化列表先于构造函数的函数体执行;
}
}; int main()
{
Test t; return ;
}

6,类中的 const 成员:

1,类中的 const 成员会被分配空间的;

1,const 成员分配的空间和我们整个类对象分配的空间一致;

2,类中的 const 成员的本质是只读变量;

1,根据编译提供的错误 bug 信息而得来;

3,类中的 oonst 成员只能在初始化列表中指定初始值;

1,编译器无法直接得到 const 成员的初始值,因此无法进入符号表成为真正意义上的常量;

1,因为只能在运行时才来定义对象,调用构造函数,继而来调用初始化列表初始化成员变量;

7,只读成员变量编程实验:

  1,代码示例:

 #include <stdio.h>

 class Value
{
private:
int mi;
public:
Value(int i)
{
printf("i = %d\n", i);
mi = i;
}
int getI()
{
return mi;
}
}; class Test
{
private:
const int ci; Value m2;
Value m3;
Value m1; public:
Test() : m1(), m2(), m3(), ci()
{
printf("Test::Test()\n");
} int getCI()
{
return ci;
} int setCI(int v)
{
int* p = const_cast<int*>(&ci); // 通过指针来操作常量对象; *p = v;
}
}; int main()
{
Test t; printf("t.ci = %d\n", t.getCI()); t.setCI(); printf("t.ci = %d\n", t.getCI()); return ;
}

2,这个实验说明:

1,类中的 const 成员不是真正意义上的常量,它只是只读变量(编译器告诉的);

8,小插曲:

1,初始化与赋值不同:

1,初始化:对正在创建的对象进行初始值设置;

1,初始化的时候,对象还没创建好,在创建的同时,我们将它的值确定了;

2,赋值:对已经存在的对象进行值设置;

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

9,小结:

1,类中可以使用初始化列表对成员进行初始化;

1,类中不能直接初始化成员变量(不论变量为一般的还是类的对象),只能通过初始化列表来初始化;

2,初始化列表先于构造函数体执行;

3,类中可以定义 const 成员变量(这里是变量);

1,const 作用于类的成员后得到的仅是只读变量;

4,const 成员变量必须在初始化列表中指定初始值;

5,const 成员变量为只读变量;

C++中初始化列表的使用的更多相关文章

  1. c++中初始化列表的初始化变量顺序问题

    例题来看:请问下面程序打印出的结果是什么? #include <iostream> #include <string> using namespace std; class b ...

  2. C++中初始化列表的使用(总结)

    原文链接 https://www.cnblogs.com/dishengAndziyu/p/10906081.html 参考链接:https://www.cnblogs.com/laiqun/p/57 ...

  3. C++类的成员初始化列表的相关问题

    在以下四中情况下,要想让程序顺利编译,必须使用成员初始化列表(member initialization list): 1,初始化一个引用成员(reference member): 2,初始化一个常量 ...

  4. C++中使用初始化列表的情况

    http://blog.csdn.net/iceshirley/article/details/5688696 要理解这个问题,从概念上,我们要知道一点,那就是构造函数的执行过程会分成两个阶段:隐式或 ...

  5. 【c++】必须在类初始化列表中初始化的几种情况

    转自:http://www.cnblogs.com/kaituorensheng/p/3477630.html 1. 类成员为const类型 2. 类成员为引用类型 #include <iost ...

  6. C++-什么时候需要在类的构造函数中使用初始化列表

    1,如果基类没有default构造函数,则意味着其不能自己初始化.如果其被派生,派生类的构造函数要负责调用基类的构造函数,并传递给它需要的参数.下例中Base 2,如果类成员没有默认构造函数.下例中E ...

  7. Effective C++学习笔记:初始化列表中成员列出的顺序和它们在类中声明的顺序相同

    类成员的默认初始化顺序是按照声明顺序进行, 如果使用初始化列表初始化成员变量, 则必须按照成员变量的声明顺序进行; 否则, 在变量之间交替赋值时, 会产生, 未初始化的变量去赋值其他变量; 同时GCC ...

  8. C++中的初始化列表中可以对那些变量或对象进行初始化

    构造函数与其函数体之间可以添加初始化列表,能对某些对象进行初始化.格式为 类名() : 变量1(参数1),变量2(参数2) { } 1.     父类的对象的构造必须在初始化列表中,如: 子类名(): ...

  9. C++中为什么构造函数初始化列表

    已经有个构造函数负责初始化,为什么还需要构造函数初始化表呢? 在以下三种情况下需要使用初始化成员列表:一,需要初始化的数据成员是对象的情况:二,需要初始化const修饰的类成员:三,需要初始化引用成员 ...

随机推荐

  1. JSON:结构化数据格式

    JSON是javascript的子类,也是作为更好的互联网传输结构化数据格式逐渐取代XML,因此要理解JSON,重要的是理解它是一种数据格式,不是一种编程语言. 语法 //javascript var ...

  2. Pjsip Porting to Hisilicon SOC

    1)HiSilicon Compiler arm-himix100-linux.tgz or arm-himix100-linux.tgz #Installation instructions are ...

  3. JAVA- 内部类及匿名内部类

    普通类,我们平时见到的那种类,就是一个后缀为.java的文件中,直接定义的类,比如 public Cat{ private String name; private int age; } 内部类, 内 ...

  4. 抽象类(abstract)与接口(interface)的区别

    如果一个类中包含抽象方法,那么这个类就是抽象类.abstract只能用来修饰类或者方法,不能用来修饰属性. 接口是指一个方法的集合,接口中的所有方法都没有方法体.接口通过关键字interface实现. ...

  5. Python错误提示:[Errno 24] Too many open files的分析与解决

    背景 最近在工作中发现了一个错误,在执行多线程扫描脚本的时候频繁出现下面这个错误 HTTPConnectionPool(host=‘t.tips', port=80): Max retries exc ...

  6. ltp-ddt nand_mtd_dd_rw_jffs2

    error: 由于在uboot下没有发现坏块,将核心代码剥离出来调试: flash_eraseall -q -j /dev/mtd1mkdir -p /mnt/partition_nand_1419m ...

  7. day19 python模块 json模块 pickle模块

    day19 python   一.序列化模块     序列类型: 列表 字符串 元组 bytes     序列化: 特指字符串和bytes, 就是把其他的数据类型转化成序列的数据类型的过程 dic = ...

  8. Spring MVC 跳转失败,但配置正确填坑

    1:正确写法 @RequestMapping("{type_key}.html") public String geren(Model model, @PathVariable S ...

  9. JSON 简单例子

    代码: json [ { "title" : "a", "num" : 1 }, { "title" : "b ...

  10. jmeter 参数化3_User Defined Variables(用户自定义变量)

    User Defined Variables:  一般用于Test Plan中不需要随请求迭代的参数设置,如:Host.Port Number 操作路径:Thread Group-->Add-- ...