C++ 参考手册
- C++11
- C++14
- C++17
- C++20
- C++ 编译器支持情况表
- 独立与宿主实现
- C++ 语言
- 变量模板(C++14 起)
- 整数字面量
- 聚合初始化
- 比较运算符
- 默认比较(C++20 起)
- 转义序列
- for 循环
- while 循环
- 用户定义转换
- SFINAE
- 主函数
- ASCII 码表
- 标识符
- 类型
- 内存模型
- 对象
- 基本概念
- 表达式
- 声明
- 初始化
- 函数
- 语句
- 类
- 运算符重载
- 模板
- 异常
- 事务性内存
- 占位符类型说明符 (C++11 起)
- decltype 说明符
- 函数声明
- final 说明符 (C++11 起)
- override 说明符(C++11 起)
- 引用声明
- 移动构造函数
- 移动赋值运算符
- 枚举声明
- constexpr 说明符(C++11 起)
- 列表初始化 (C++11 起)
- 构造函数与成员初始化器列表
- using 声明
- nullptr,指针字面量
- 基础类型
- 类型别名,别名模版 (C++11 起)
- 形参包
- 联合体声明
- 字符串字面量
- 用户定义字面量 (C++11 起)
- 属性说明符序列(C++11 起)
- Lambda 表达式 (C++11 起)
- noexcept 说明符 (C++11 起)
- noexcept 运算符 (C++11 起)
- alignof 运算符(C++11 起)
- alignas 说明符 (C++11 起)
- 存储类说明符
- 基于范围的 for 循环 (C++11 起)
- static_assert 声明
- 隐式转换
- 代用运算符表示
- 自增/自减运算符
- 折叠表达式(C++17 起)
- 类模板实参推导(C++17 起)
- 模板形参与模板实参
- if 语句
- inline 说明符
- 结构化绑定声明 (C++17 起)
- switch 语句
- 字符字面量
- 命名空间
- 求值顺序
- 复制消除
- consteval 说明符 (C++20 起)
- constinit 说明符 (C++20 起)
- 协程 (C++20)
- 模块 (C++20 起)
- 约束与概念 (C++20 起)
- new 表达式
- do-while 循环
- continue 语句
- break 语句
- goto 语句
- return 语句
- 动态异常说明
- throw 表达式
- try 块
- 命名空间别名
- 类声明
- cv(const 与 volatile)类型限定符
- 默认初始化
- 值初始化(C++03 起)
- 零初始化
- 复制初始化
- 直接初始化
- 常量初始化
- 引用初始化
- 值类别
- C++ 运算符优先级
- 布尔字面量
- 浮点字面量
- typedef 说明符
- 显式类型转换
- static_cast 转换
- dynamic_cast 转换
- const_cast 转换
- reinterpret_cast 转换
- delete 表达式
- 构造函数与成员初始化器列表
- this 指针
- 访问说明符
- 友元声明
- virtual 函数说明符
- explicit 说明符
- 静态成员
- 默认构造函数
- 复制构造函数
- 复制赋值运算符
- 析构函数
- 类模板
- 函数模板
- 显式(全)模板特化
- 汇编声明
- C++ 的历史
- 作用域
- 生存期
- 定义与单一定义规则(ODR)
- 名字查找
- 有限定的名字查找
- 无限定的名字查找
- 如同规则
- 未定义行为
- 翻译阶段
- 常量表达式
- 赋值运算符
- 算术运算符
- 逻辑运算符
- 成员访问运算符
- 其他运算符
- sizeof 运算符
- typeid 运算符
- 指针声明
- 数组声明
- 语言链接
- 详述类型说明符
- 默认实参
- 变长实参
- 实参依赖查找
- 重载决议
- 重载函数的地址
- 注入类名
- 非静态数据成员
- 非静态成员函数
- 嵌套类
- 派生类
- 空基类优化
- 抽象类
- 位域
- 转换构造函数
- 成员模板
- 模板实参推导
- 部分模板特化
- sizeof... 运算符
- 待决名
- 函数 try 块
- 扩充命名空间 std
- 字母缩写
- RAII
- 三/五/零之法则
- PImpl
- 零开销原则
- 类型
- 隐式转换
- 注释
- C++ 关键词
- 预处理器
- C++ 标准库头文件
- 具名要求
- 功能特性测试 (C++20)
- 工具库
- 类型支持(基本类型、RTTI、类型特性)
- 概念库 (C++20)
- 错误处理
- 动态内存管理
- 日期和时间工具
- 字符串库
- 容器库
- 迭代器库
- 范围库 (C++20)
- 算法库
- 数值库
- 输入/输出库
- 文件系统库
- 本地化库
- 正则表达式库
- 原子操作库
- 线程支持库
- 实验性 C++ 特性
- 有用的资源
- 索引
- std 符号索引
- 协程支持 (C++20)
- C++ 关键词
直接初始化
从明确的构造函数实参的集合初始化对象。
语法
T object ( arg );
T object |
(1) | ||||||||
T object { arg };
|
(2) | (C++11 起) | |||||||
T ( other )
T |
(3) | ||||||||
static_cast< T >( other )
|
(4) | ||||||||
new T( args, ...)
|
(5) | ||||||||
Class:: Class() : member( args, ...) { ... }
|
(6) | ||||||||
[ arg](){ ... }
|
(7) | (C++11 起) | |||||||
解释
在下列场合进行直接初始化:
1) 以表达式或花括号初始化器列表 (C++11 起)的非空带括号列表初始化
2) 作为列表初始化的一部分,以花括号环绕的单个初始化器初始化非类类型和非数组类型对象时,以初始化提供了初始化器的元素(注意:对于类类型和其他使用花括号初始化器列表的初始化,见列表初始化)
5) 用带有非空初始化器的 new 表达式初始化决议动态存储期的对象
6) 用构造函数初始化器列表初始化基类或非静态成员
7) 在 lambda 表达式中从按复制捕捉的变量初始化闭包对象的成员
直接初始化的效果是:
- 若
T
是数组类型,则
|
(C++20 前) |
struct A { explicit A(int i = 0) {} }; A a[2](A(1)); // OK:以 A(1) 初始化 a[0] 并以 A() 初始化 a[1] A b[2]{A(1)}; // 错误:从 {} 隐式复制初始化 a[1] 选择了 explicit 构造函数 |
(C++20 起) |
- 若
T
是类类型,
(C++17 起) |
- 检验
T
的构造函数并由重载决议选取最佳匹配。然后调用该构造函数初始化对象。
- 检验
struct B { int a; int&& r; }; int f(); int n = 10; B b1{1, f()}; // OK:延长生存期 B b2(1, f()); // 良构,但有悬垂引用 B b3{1.0, 1}; // 错误:窄化转换 B b4(1.0, 1); // 良构,但有悬垂引用 B b5(1.0, std::move(n)); // OK |
(C++20 起) |
- 否则,若
T
是非类类型但源类型是类类型,则检验源类型及其各基类的转换函数,并由重载决议选取最佳匹配。然后用选取的用户定义转换,转换初始化器表达式为所初始化的对象。 - 否则,若
T
为 bool 而原类型是 std::nullptr_t,则被初始化对象的值为 false。 - 否则,使用标准转换(若有必要),转换 other 的值为
T
的无 cv 限定版本,而所初始化的对象的初值为(可能为转换后的)该值。
注解
直接初始比复制初始化更宽容:复制初始化仅考虑非 explicit 构造函数和非 explicit 的用户定义转换函数,而直接初始化考虑所有构造函数和所有用户定义转换函数。
在使用直接初始化语法 (1)(带圆括号)的变量声明和函数声明之间有歧义的情况下,编译器始终选择函数声明。此消歧义规则有时是反直觉的,并且已被称为最烦人的分析。
#include <iterator> #include <string> #include <fstream> int main() { std::ifstream file("data.txt"); // 下面是函数声明: std::string str(std::istreambuf_iterator<char>(file), std::istreambuf_iterator<char>()); // 它声明名为 str 的函数,其返回类型为 std::string, // 第一参数拥有 std::istreambuf_iterator<char> 类型和名称 "file" // 第二参数无名称并拥有类型 std::istreambuf_iterator<char>(), // 它被重写成函数指针类型 std::istreambuf_iterator<char>(*)() // C++11 前的修正:环绕实参之一的额外括号 std::string str( (std::istreambuf_iterator<char>(file) ), std::istreambuf_iterator<char>()); // C++11 后的修正:任何实参的列表初始化 std::string str(std::istreambuf_iterator<char>{file}, {}); }
类似地,在以函数式转型表达式 (3) 为其最左子表达式的表达式语句,和声明语句间有歧义的情况下,以将它当做声明解决歧义。此消歧义是纯语法的:它不考虑语句中出现的名字除了是否为类型名之外的含义。
struct M { }; struct L { L(M&); }; M n; void f() { M(m); // 声明,等价于 M m; L(n); // 非良构的声明 L(l)(m); // 仍然是声明 }
示例
运行此代码
#include <string> #include <iostream> #include <memory> struct Foo { int mem; explicit Foo(int n) : mem(n) {} }; int main() { std::string s1("test"); // 自 const char* 的构造函数 std::string s2(10, 'a'); std::unique_ptr<int> p(new int(1)); // OK:允许 explicit 构造函数 // std::unique_ptr<int> p = new int(1); // 错误:构造函数为 explicit Foo f(2); // f 被直接初始化: // 构造函数形参 n 从右值 2 复制初始化 // f.mem 从形参 n 直接初始化 // Foo f2 = 2; // 错误:构造函数为 explicit std::cout << s1 << ' ' << s2 << ' ' << *p << ' ' << f.mem << '\n'; }
输出:
test aaaaaaaaaa 1 2