直接初始化

来自cppreference.com
< cpp‎ | language

从构造函数参数的显式集合初始化对象。

目录

[编辑] 语法

T object ( arg );

T object ( arg1, arg2, ... );

(1)
T object { arg }; (2) (C++11 起)
T ( other )

T ( arg1, arg2, ... )

(3)
static_cast< T >( other ) (4)
new T(args, ...) (5)
Class::Class() : member(args, ...) { ... } (6)
[arg](){ ... } (7) (C++11 起)

[编辑] 解释

在下列场合进行直接初始化:

1) 以表达式或花括号初始化器列表 (C++11 起)的非空带括号列表初始化
2) 以单个花括号环绕的初始化器初始化非类类型对象(注意:对于类类型和其他使用花括号初始化器列表的初始化,见列表初始化
3)函数转型或以带括号表达式列表初始化纯右值临时量
4)static_cast 表达式初始化纯右值临时量
5) 用带非空初始化器的 new 表达式初始化拥有动态存储期的对象
6) 用构造函数初始化器列表初始化基类或非静态成员
7) 在 lambda 表达式中从以复制捕捉的变量初始化闭包对象成员

直接初始化的效果是:

  • T 是类类型,
  • 若初始化器为纯右值表达式,而其类型与 T 为相同的类(忽略 cv 限定),则用初始化器表达式自身,而非从它实质化的临时量,初始化目标对象:参阅复制消除
(C++17 起)
  • 检验 T 的构造函数并由重载决议选取最佳匹配。然后调用该构造函数初始化对象。
  • 否则,若 T 是非类类型但源类型是类类型,则检验源类型与其基类的转换函数,并由重载决议选取最佳匹配。然后用选取的用户定义转换,转换初始化器表达式为正在初始化的对象。
  • 否则,若 Tbool 而原类型是 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

[编辑] 参阅