命名空间
变体
操作

直接初始化

来自 cppreference.com
< cpp‎ | language
 
 
C++ 语言
表达式
替代表示
字面量
布尔型 - 整数型 - 浮点型
字符型 - 字符串型 - nullptr (C++11)
用户定义 (C++11)
实用程序
属性 (C++11)
类型
typedef 声明
类型别名声明 (C++11)
强制转换
内存分配
特定于类的函数属性
explicit (C++11)
static

特殊成员函数
模板
其他
 
 

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

内容

[编辑] 语法

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) 使用单个大括号括起来的初始化器初始化非类类型对象(注意:对于类类型和大括号括起来的初始化列表的其他用法,请参阅 列表初始化)(自 C++11 起)
3) 通过右值临时对象(直到 C++17)右值的结果对象(自 C++17 起)使用函数风格强制转换或带括号的表达式列表进行初始化。
4) 通过右值临时对象(直到 C++17)右值的结果对象(自 C++17 起)使用static_cast 表达式进行初始化。
5) 使用带有初始化器的 new 表达式初始化具有动态存储期限的对象。
6) 使用构造函数初始化列表初始化基类或非静态成员。
7) 从 Lambda 表达式中通过复制捕获的变量初始化闭包对象成员。

直接初始化的效果是

  • 如果 T 是数组类型,
  • 程序格式错误。
(直到 C++20)
struct A
{
    explicit A(int i = 0) {}
};
 
A a[2](A(1)); // OK: initializes a[0] with A(1) and a[1] with A()
A b[2]{A(1)}; // error: implicit copy-list-initialization of b[1]
              //        from {} selected explicit constructor
(自 C++20 起)
  • 如果 T 是类类型,
  • 如果初始化器是类型与 T 相同的右值表达式(忽略 cv 限定符),则初始化器表达式本身(而不是从其生成的临时对象)用于初始化目标对象。
    (在 C++17 之前,编译器可能在此情况下省略了从右值临时对象进行的构造,但合适的构造函数仍然必须可访问:请参阅 复制省略)
(自 C++17 起)
  • 检查T的构造函数,并通过重载解析选择最佳匹配。然后调用该构造函数来初始化对象。
  • 否则,如果目标类型是(可能经过 cv 限定的)聚合类,则按照 聚合初始化 中所述进行初始化,但允许进行窄化转换,不允许使用指定初始化器,绑定到引用的临时对象不会延长其生命周期,没有大括号省略,并且所有没有初始化器的元素都将进行 值初始化
struct B
{
    int a;
    int&& r;
};
 
int f();
int n = 10;
 
B b1{1, f()};            // OK, lifetime is extended
B b2(1, f());            // well-formed, but dangling reference
B b3{1.0, 1};            // error: narrowing conversion
B b4(1.0, 1);            // well-formed, but dangling reference
B b5(1.0, std::move(n)); // OK
(自 C++20 起)
  • 否则,如果T是非类类型,但源类型是类类型,则检查源类型及其基类的(如果有)转换函数,并通过重载解析选择最佳匹配。然后使用选定的用户定义转换将初始化表达式转换为正在初始化的对象。
  • 否则,如果Tbool 并且源类型是 std::nullptr_t,则初始化对象的值为 false
  • 否则,使用 标准转换(如果需要)将 other 的值转换为T的 cv 未限定版本,并且正在初始化对象的初始值是(可能已转换的)值。

[编辑] 说明

直接初始化比复制初始化更宽松:复制初始化只考虑非 显式 构造函数和非显式用户定义的 转换函数,而直接初始化则考虑所有构造函数和所有用户定义的转换函数。

在使用直接初始化语法 (1)(带圆括号)的变量声明与 函数声明 之间存在歧义的情况下,编译器始终选择函数声明。这种消除歧义规则有时会违反直觉,被称为 最令人困惑的解析

#include <fstream>
#include <iterator>
#include <string>
 
int main()
{
    std::ifstream file("data.txt");
 
    // The following is a function declaration:
    std::string foo1(std::istreambuf_iterator<char>(file),
                     std::istreambuf_iterator<char>());
    // It declares a function called foo1, whose return type is std::string,
    // first parameter has type std::istreambuf_iterator<char> and the name "file",
    // second parameter has no name and has type std::istreambuf_iterator<char>(),
    // which is rewritten to function pointer type std::istreambuf_iterator<char>(*)()
 
    // Pre-C++11 fix (to declare a variable) - add extra parentheses around one
    // of the arguments:
    std::string str1((std::istreambuf_iterator<char>(file)),
                      std::istreambuf_iterator<char>());
 
    // Post-C++11 fix (to declare a variable) - use list-initialization for any
    // of the arguments:
    std::string str2(std::istreambuf_iterator<char>{file}, {});
}

[编辑] 示例

#include <iostream>
#include <memory>
#include <string>
 
struct Foo
{
    int mem;
    explicit Foo(int n) : mem(n) {}
};
 
int main()
{
    std::string s1("test"); // constructor from const char*
    std::string s2(10, 'a');
 
    std::unique_ptr<int> p(new int(1));  // OK: explicit constructors allowed
//  std::unique_ptr<int> p = new int(1); // error: constructor is explicit
 
    Foo f(2); // f is direct-initialized:
              // constructor parameter n is copy-initialized from the rvalue 2
              // f.mem is direct-initialized from the parameter n
//  Foo f2 = 2; // error: constructor is explicit
 
    std::cout << s1 << ' ' << s2 << ' ' << *p << ' ' << f.mem  << '\n';
}

输出

test aaaaaaaaaa 1 2

[编辑] 参见