• std::initializer_list详解


    std::initializer_list介绍

    initializer_list是C++11提供的一种新类型,其定义于头文件中,此头文件是工具库的一部分,

    定义如下:

    1. namespace std {
    2. template<class E> class initializer_list {
    3. public:
    4. using value_type = E;
    5. using reference = const E&;
    6. using const_reference = const E&;
    7. using size_type = size_t;
    8. using iterator = const E*;
    9. using const_iterator = const E*;
    10. constexpr initializer_list() noexcept;
    11. constexpr size_t size() const noexcept; // 元素数量
    12. constexpr const E* begin() const noexcept; // 首元素
    13. constexpr const E* end() const noexcept; // 末元素后一位置
    14. };
    15. // initializer_list 范围访问
    16. template<class E> constexpr const E* begin(initializer_list<E> il) noexcept;
    17. template<class E> constexpr const E* end(initializer_list<E> il) noexcept;
    18. }

     

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

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

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

    initializer_list 可由一对指针或指针与其长度实现。复制一个 std::initializer_list 不会复制其底层对象。

    底层数组不保证在原始 initializer_list 对象的生存期结束后继续存在。 std::initializer_list 的存储是未指定的(即它可以是自动、临时或静态只读内存,依赖场合)。 (C++14 前)

    底层数组是 const T[N] 类型的临时数组,其中每个元素都从原始初始化器列表的对应元素复制初始化(除非窄化转换非法)。底层数组的生存期与任何其他临时对象相同,除了从数组初始化 initializer_list 对象会延长数组的生存期,恰如绑定引用到临时量(有例外,例如对于初始化非静态类成员)。底层数组可以分配在只读内存。 (C++14 起)

    若声明了 std::initializer_list 的显式或偏特化则程序为病式。

    1.成员类型

    成员类型定义
    value_typeT
    referenceconst T&
    const_referenceconst T&
    size_typestd_size_t
    iteratorconst T*
    const_iteratorconst T*

    2.成员函数

    构造函数

    initializer_list() noexcept;(C++11 起) (C++14 前)
    constexpr initializer_list() noexcept;(C++14 起)

     

    1. #include <iostream>
    2. #include <initializer_list>
    3. int main(){
    4. std::initializer_list<int> empty_list;
    5. std::cout << "empty_list.size(): " << empty_list.size() << '\n';
    6. // 用列表初始化创建初始化器列表
    7. std::initializer_list<int> digits {1, 2, 3, 4, 5};
    8. std::cout << "digits.size(): " << digits.size() << '\n';
    9. // auto 的特殊规则表示‘ fraction '拥有类型
    10. // type std::initializer_list
    11. auto fractions = {3.14159, 2.71828};
    12. std::cout << "fractions.size(): " << fractions.size() << '\n';
    13. }

     

    结果如下:

    • size :返回initializer_list中元素数目
    • begin:返回指向首元素的指针
    • end:返回指向末尾元素后一位置的指针

    3.非成员函数

    std::begin(std::initializer_list) (C++11)特化 std::begin
    std::end(std::initializer_list)(C++11) 定义于头文件特化std::end
    rbegin(std::initializer_list) (C++14)特化std::rbegin
    rend(std::initializer_list)(C++14)特化std::rend

     

    1. #include <iostream>
    2. #include <vector>
    3. #include <initializer_list>
    4. template <class T>
    5. struct S {
    6. std::vector<T> v;
    7. S(std::initializer_list<T> l) : v(l) {
    8. std::cout << "constructed with a " << l.size() << "-element list\n";
    9. }
    10. void append(std::initializer_list<T> l) {
    11. v.insert(v.end(), l.begin(), l.end());
    12. }
    13. std::pair<const T*, std::size_t> c_arr() const {
    14. return {&v[0], v.size()}; //return 语句中复制列表初始化
    15. // 这不使用 std::initializer_list
    16. }
    17. };
    18. template <typename T>
    19. void templated_fn(T) {}
    20. int main(){
    21. S<int> s = {1, 2, 3, 4, 5}; // 复制初始化
    22. s.append({6, 7, 8}); // 函数调用中的列表初始化
    23. std::cout << "The vector size is now " << s.c_arr().second << " ints:\n";
    24. for (auto n : s.v)
    25. std::cout << n << ' ';
    26. std::cout << '\n';
    27. std::cout << "Range-for over brace-init-list: \n";
    28. for (int x : {-1, -2, -3}) // auto 的规则令此带范围 for 工作
    29. std::cout << x << ' ';
    30. std::cout << '\n';
    31. auto al = {10, 11, 12}; // auto 的特殊规则
    32. std::cout << "The list bound to auto has size() = " << al.size() << '\n';
    33. // templated_fn({1, 2, 3}); // 编译错误!“ {1, 2, 3} ”不是表达式,
    34. // 它无类型,故 T 无法推导
    35. templated_fn<std::initializer_list<int>>({1, 2, 3}); // OK
    36. templated_fn<std::vector<int>>({1, 2, 3}); // 也 OK
    37. }

    结果如下:

  • 相关阅读:
    带头双向循环链表讲解-思路清晰+画图+代码实现
    2022年最新青海建筑八大员(标准员)考试试题及答案
    Spring Boot Security自带功能齐全的HttpFirewall防火墙
    10道小学趣味数学题
    学内核之九:学会偷懒,善用内核的调试日志
    中标麒麟操作系统网络应用服务部署与实现
    CAN和CANFD通信介绍
    zsh和ohmyzsh安装指南+插件推荐
    无线数字平板探测器维修Mars1717XU-VSI故障分析
    蒙特卡洛估计举例
  • 原文地址:https://blog.csdn.net/hulinhulin/article/details/133444819