std::initializer_list

来自cppreference.com
< cpp‎ | utility
 
 
工具库
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)
(C++20)
swap 与类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)
 
 

(勿与成员初始化器列表混淆)

在标头 <initializer_list> 定义
template< class T >
class initializer_list;
(C++11 起)

std::initializer_list<T> 类型对象是一个访问 const T 类型对象数组的轻量代理对象。

std::initializer_list 对象在这些时候自动构造:

  • 花括号初始化器列表 列表初始化一个对象,其中对应构造函数接受一个 std::initializer_list 参数
  • 花括号初始化器列表 为赋值的右运算数,或函数调用参数,而对应的赋值运算符/函数接受 std::initializer_list 参数
  • 绑定花括号初始化器列表 到 auto,包括在范围 for 循环

std::initializer_list 可由一对指针或指针与其长度实现。复制一个 std::initializer_list 不会复制它对应的初始化器列表的基底数组

如果声明了 std::initializer_list 的显式或偏特化,那么程序非良构。

成员类型

成员类型 定义
value_type T
reference const T&
const_reference const T&
size_type std::size_t
iterator const T*
const_iterator const T*

成员函数

创建空的初始化器列表
(公开成员函数)
容量
返回初始化器列表中的元素数目
(公开成员函数)
迭代器
返回指向首元素的指针
(公开成员函数)
返回指向末元素后一位置的指针
(公开成员函数)

非成员函数

特化 std::begin
(函数模板)
特化 std::end
(函数模板)
std::initializer_list 重载的自由函数模板
返回指向一个容器或数组的逆向迭代器
(函数模板)
(C++14)
返回容器或数组的逆向尾迭代器
(函数模板)
(C++17)
检查容器是否为空
(函数模板)
(C++17)
获得指向底层数组的指针
(函数模板)

示例

#include <initializer_list>
#include <iostream>
#include <vector>
 
template<class T>
struct S
{
    std::vector<T> v;
 
    S(std::initializer_list<T> l) : v(l)
    {
         std::cout << "以包含 " << l.size() << " 个元素的列表构造\n";
    }
 
    void append(std::initializer_list<T> l)
    {
        v.insert(v.end(), l.begin(), l.end());
    }
 
    std::pair<const T*, std::size_t> c_arr() const
    {
        return {&v[0], v.size()}; // 在 return 语句中进行复制列表初始化
                                  // 没有使用 std::initializer_list
    }
};
 
template<typename T>
void templated_fn(T) {}
 
int main()
{
    S<int> s = {1, 2, 3, 4, 5}; // 复制列表初始化
    s.append({6, 7, 8});        // 在函数调用中进行列表初始化
 
    std::cout << "现在 vector 含有 " << s.c_arr().second << " 个 int:\n";
 
    for (auto n : s.v)
        std::cout << n << ' ';
    std::cout << '\n';
 
    std::cout << "用范围 for 遍历花括号初始化器列表:\n";
 
    for (int x : {-1, -2, -3}) // 对 auto 的规则使得此范围 for 有效
        std::cout << x << ' ';
    std::cout << '\n';
 
    auto al = {10, 11, 12}; // 对 auto 的特殊规则
 
    std::cout << "绑定到 auto 的列表的 size() = " << al.size() << '\n';
 
//  templated_fn({1, 2, 3}); // 编译错误!"{1, 2, 3}" 不是表达式,
                             // 它没有类型,所以不能推导出 T
    templated_fn<std::initializer_list<int>>({1, 2, 3}); // OK
    templated_fn<std::vector<int>>({1, 2, 3});           // 同样 OK
}

输出:

以包含 5 个元素的列表构造
现在 vector 含有 8 个 int:
1 2 3 4 5 6 7 8
用范围 for 遍历花括号初始化器列表:
-1 -2 -3 
绑定到 auto 的列表的 size() = 3

缺陷报告

下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。

缺陷报告 应用于 出版时的行为 正确行为
LWG 2129 C++11 可以显式特化或偏特化 std::initializer_list 此时程序非良构

参阅

(C++20)
对象的连续序列上的无所有权视图
(类模板)
只读的字符串视图
(类模板)