C++的学习中,我想每个人都被变量定义和申明折磨过,比如我在大学笔试过的几家公司,都考察了const和变量,类型的不同排列组合,让你区别有啥不同。反正在学习C++过程中已经被折磨惯了,今天再来看看重温下那段“辉煌的历史”。先来看一段代码:
Player pa; // (a) Player pb(); // (b) Player pc = Player(); // (c) Player pd(Player()); // (d) pd = Player() // (e)
a,b,c,d 都是申明一个变量,a 很容易理解就是申明一个变量,b第一感觉是函数申明,其实不是,而是对象申明。对于c以为是调用了operator= 赋值运算吧?完全不是,而是先生成一个对象,然后调用Player的拷贝构造函数,生成对象pc。d和c是一样的。e才是真正调用赋值操作。是不是已经被这各式各样的对象申明搞的晕头专向了。现在C++11初始化列表正式登场。
二、简介
在看C++11初始化前,先来回忆一下C语言中的结构体初始化,代码如下:
#include <iostream> struct Player{ int id; const char* name; }; int main() { Player player = {10001, "c++"}; printf("%d, %s\n", player.id, player.name); }
结构体变量可以列表初始化,非常方便。C++11引入了初始化列表来初始化变量和对象。
三、如何使用
1、系统内置类型
int ia{1}; // (a) int ib = {1}; // (b) int ic(1); // (c) int id = 1; // (d)很明显,还是d 更符合习惯。
std::vector<int> va{1, 2, 3}; // (a) std::vector<int> vb = {1, 2, 3}; // (b) std::vector<int> vc(1, 10); // (c) std::vector<int> vd{1, 10}; // (d)
通过初始化列表可以弥补c中只能初始化相同数字的问题。在使用中c和d不要混淆了。
std::pair<int, const char*> getPlayer() { return {10001, "c++"}; } std::map<int, const char*> players = {{10001, "c++"}, {10002, "java”}};
还可以返回pair类型,初始化map都是可以的。
2、自定义类型
对于单个参数初始化,类型匹配构造函数,不需要自定义构造函数。
Player pa{}; // (a) Player pb; // (b) Player pc(); // (c) Player pd(b); // (d) Player pe = b; // (e) Player pf = {b}; // (f)
对于上面几种变量初始化,推荐a, 如果是带参数的构造函数,推荐b,若果是不带参数的构造函数。
3、如果是自己想实现初始化列表构造函数,拷贝函数,赋值函数,需要包含initializer_list 这个头文件。
class MyClass{ public: MyClass(int a):a_(a){ std::cout << "normal initializer list\n"; } MyClass(std::initializer_list<int> a):b_(a) { std::cout << "initializer list constructor\n"; } MyClass(MyClass& my) { std::cout << "copy constructor\n"; this->a_ = my.a_; this->b_ = my.b_; } MyClass& operator=(MyClass& my) { std::cout << "operator = constructor\n"; this->a_ = my.a_; this->b_ = my.b_; return *this; } private: int a_; std::initializer_list<int> b_; };
这是自定义类,带有初始化列表构造函数的类,下面来练习下学过的类相关C++知识:
MyClass ma{1}; // (a) MyClass mb = {1, 2, 3}; // (b) MyClass mc(2); // (c) MyClass md = b; // (d) MyClass me(c); // (e) MyClass mf{e}; // (f) auto l{2, 2, 3,3}; MyClass mh{l}; // (e) ma = mb; // (h)
看一下每次调用都输出什么结果,答案在这里:
initializer list constructor
initializer list constructor
normal constructor list
copy constructor
copy constructor
copy constructor
initializer list constructor
operator = constructor
四、为啥需要初始化列表
1、避免类申明对象混淆,区分对待,对于C++为啥有小括号初始化对象这一说,大家可以自行google,看来不管是谁都有犯错误的时候,勇于承认错误还是好同志。
2、在初始化多个变量时方便
3、避免数据切割,因为通过初始化列表是不允许隐式转换的,相关知识可以参考我的这篇文章。
五、注意事项
1、在申明变量的时候,少用小括号,程序可读性更高
2、使用初始化列表防止隐式转换,减少bug
3、通过 {} 返回的对象是const类型,不可转换