Cpp11 列表初始化
从花括号初始化器列表初始化对象
直接列表初始化
语法
直接列表初始化
T object { arg1, arg2, ... };
T { arg1, arg2, ... }
new T { arg1, arg2, ... }
Class { T member { arg1, arg2, ... }; };
Class::Class() : member{arg1, arg2, ...} {...}//在构造函数的成员初始化列表中,若使用花括号初始化器列表
实例
class T {
public:
vector<int> veca;
vector<int> vecb;
vector<int> vecc;
//不能有自定义的构造函数。
};
class B {
T t{{1, 2}, {2, 3}, {3, 4}};
};
int main() {
T t{ {1, 2}, {2, 3}, {3, 4} };
auto t1 = T{ {1, 2}, {2, 3}, {3, 4} };
auto *p = new T{ {1, 2}, {2, 3}, {3, 4} };
return 0;
}
复制列表初始化
复制列表初始化
T object = {arg1, arg2, ...}; //赋值运算
function( { arg1, arg2, ... } ) //函数参数
return { arg1, arg2, ... } ; //函数返回值
object[ { arg1, arg2, ... } ] //在具有用户定义的 operator[] 的下标表达式中,以列表初始化对重载运算符的形参初始化
object = { arg1, arg2, ... }
U( { arg1, arg2, ... } )
Class { T member = { arg1, arg2, ... }; }; // 在使用等号的非静态数据成员初始化器中
列表初始化规则
聚合类型可直接列表初始化。
聚合类型
- 普通数组
–char chs[10], int ints[10]
- 类
– 没有自定义构造函数– 没有私有或被保护的成员– 没有基类– 没有虚函数– 没有{}
和=
直接初始化的非静态数据成员
代码实例
#include <iostream>
#include <vector>
#include <map>
#include <string>
struct Foo {
std::vector<int> mem = {1,2,3}; // 非静态成员的列表初始化
std::vector<int> mem2;
Foo() : mem2{-1, -2, -3} {} // 构造函数中的成员列表初始化
};
std::pair<std::string, std::string> f(std::pair<std::string, std::string> p)
{
return {p.second, p.first}; // return 语句中的列表初始化
}
int main()
{
int n0{}; // 值初始化(为零)
int n1{1}; // 直接列表初始化
std::string s1{'a', 'b', 'c', 'd'}; // initializer_list 构造函数调用
std::string s2{s1, 2, 2}; // 常规构造函数调用
std::string s3{0x61, 'a'}; // initializer_list 构造函数偏好 (int, char)
int n2 = {1}; // 复制列表初始化
double d = double{1.2}; // 纯右值的列表初始化,然后复制初始化
auto s4 = std::string{"HelloWorld"}; // 同上, C++17 起不创建临时对象
std::map<int, std::string> m = { // 嵌套列表初始化
{1, "a"},
{2, {'a', 'b', 'c'} },
{3, s1}
};
std::cout << f({"hello", "world"}).first // 函数调用中的列表初始化
<< '\n';
const int (&ar)[2] = {1,2}; // 绑定左值引用到临时数组
int&& r1 = {1}; // 绑定右值引用到临时 int
// int& r2 = {2}; // 错误:不能绑定右值到非 const 左值引用
// int bad{1.0}; // 错误:窄化转换
unsigned char uc1{10}; // 可以
// unsigned char uc2{-1}; // 错误:窄化转换
Foo f;
std::cout << n0 << ' ' << n1 << ' ' << n2 << '\n'
<< s1 << ' ' << s2 << ' ' << s3 << '\n';
for(auto p: m)
std::cout << p.first << ' ' << p.second << '\n';
for(auto n: f.mem)
std::cout << n << ' ';
for(auto n: f.mem2)
std::cout << n << ' ';
}