这里注意,这个与构造函数的初始化列表没有任何关系。首先我们先看C++98中列表,{}的使用场景:允许数组使用列表赋值
cpp
int a[] = {1,2,3,4};
int b[5] = {0};
列表初始化
但是对于vector这样的自定义类型就无法这样赋值,而是需要循环遍历赋值。所以C++11就引入了列表初始化:
cpp
int main()
{
// 内置类型变量
int x1 = {10};
int x2{10};
int x3 = 1+2;
int x4 = {1+2};
int x5{1+2};
// 数组
int a[5] {1,2,3,4,5};
int b[]{1,2,3,4,5};
// 动态数组,在C++98中不支持
int* p1 = new int[5]{1,2,3,4,5};
// 标准容器
vector<int> v{1,2,3,4,5};
map<int, int> m{{1,1}, {2,2,},{3,3},{4,4}};
return 0;
}
自定义类如何支持列表初始化
对于自定义类型:也是支持用列表进行初始化的,但是必须要有对应参数类型和个数的构造函数,因为用{}初始化,会调用对应的构造函数。
cpp
class A
{
public:
A(int x = 0, int y = 0): _x(x), _y(y)
{}
private:
int _x;
int _y;
};
int main()
{
A a{1,2};
return 0;
}
我们要说一说,他是怎么支持stl中容器的初始化的,很显然,vector中只有capacity,size,还有一个指针,但是无论多长的列表都可以使用{}进行初始化。所以不满足上述红字的要求,那么C++11是如何支持的呢?这就要说到initializer_list,他是C++中的一个类
initializer_list

对象想要支持列表初始化,需给该类(模板类)添加一个带有initializer_list类型参数的构造函数即
可。注意:initializer_list是系统自定义的类模板,该类模板中主要有三个方法:begin()、end()迭代器以及获取区间中元素个数的方法size()。
我们来看一看这个类:
可以看到sizeof它,仅仅只有8个字节,但是给了它3个int啊,这是因为它的内部只有两个指针。你不能手动往 initializer_list 里添加或修改值 ,它是一个轻量的、只读的容器视图(view) 。唯一能让它包含元素的方式就是用 {}
初始化(调用构造函数)。
接下来说一说,形如vector这样的自定义类是如何支持{}初始化的
其实就是调用对应的构造函数。
构造函数类似:
cpp
vector(initializer_list<T> lt)
:_start(nullptr)
,_finish(nullptr)
,_endofstorage(nullptr)
{
typename initializer_list<T>::iterator it = lt.begin();
auto it = it.begin();
while(it != lt.end())
{
push_back(*it);
++it;
}
}
这种初始化是调用了一个initializer_list作为参数的构造函数来初始化:vector<int> v1 = {1,2,3,4,5};
所以,自定义类型对象可以使用{}初始化,必须要有对应参数类型和个数的构造函数,因为用{}初始化,会调用对应的构造函数。如果不是这种自定义类型,而是类似STL容器,如果要支持{}初始化,那么里面必须有支持一个initializer_list作为参数的构造函数