|
| 1 | +# std::initializer_listの配列を静的記憶域に配置する [P2752R3] |
| 2 | +* cpp26[meta cpp] |
| 3 | + |
| 4 | +<!-- start lang caution --> |
| 5 | + |
| 6 | +このページはC++26に採用される見込みの言語機能の変更を解説しています。 |
| 7 | + |
| 8 | +のちのC++規格でさらに変更される場合があるため[関連項目](#relative-page)を参照してください。 |
| 9 | + |
| 10 | +<!-- last lang caution --> |
| 11 | + |
| 12 | +## 概要 |
| 13 | +C++11で導入された[`std::initializer_list`](/reference/initializer_list/initializer_list.md)は、自作クラスで組み込み配列のような配列初期化をする構文であるが、そのメモリとしてはスタック上に配置されていた。 |
| 14 | + |
| 15 | +```cpp |
| 16 | +std::vector<int> v = {1, 2, 3}; |
| 17 | + |
| 18 | +// 以下と等価: |
| 19 | +const int __a[] = {1, 2, 3}; |
| 20 | +std::vector<int> v(__a, __a + 3); |
| 21 | +``` |
| 22 | +
|
| 23 | +C++26では、プリプロセス時にファイル読み込む[`#embed`](/lang/cpp26/embed.md)機能が導入されることもあり、スタックオーバーフローのリスクがあった。 |
| 24 | +
|
| 25 | +```cpp |
| 26 | +std::vector<char> v = { |
| 27 | + #embed "2mb-image.png" |
| 28 | +}; |
| 29 | +``` |
| 30 | + |
| 31 | +このため、C++26では、定数値のみ、かつ`mutable`に保持しない[`std::initializer_list`](/reference/initializer_list/initializer_list.md)は、静的記憶域に配置されるようになる。 |
| 32 | + |
| 33 | +```cpp |
| 34 | +void f(std::initializer_list<double> il); |
| 35 | + |
| 36 | +void g(float x) { |
| 37 | + f({1, x, 3}); |
| 38 | + |
| 39 | + // 変数を含むため、静的記憶域には保持されない。 |
| 40 | + // 以下と等価: |
| 41 | + // const double __a[3] = {double{1}, double{x}, double{3}}; |
| 42 | + // f(std::initializer_list<double>(__a, __a+3)); |
| 43 | +} |
| 44 | + |
| 45 | +void h() { |
| 46 | + f({1, 2, 3}); |
| 47 | + |
| 48 | + // 定数のみであるため、静的記憶域に保持される。 |
| 49 | + // 以下と等価: |
| 50 | + // static constexpr double __b[3] = {double{1}, double{2}, double{3}}; |
| 51 | + // f(std::initializer_list<double>(__b, __b+3)); |
| 52 | +} |
| 53 | + |
| 54 | +struct A { |
| 55 | + mutable int i; |
| 56 | +}; |
| 57 | + |
| 58 | +void q(std::initializer_list<A>); |
| 59 | + |
| 60 | +void r() { |
| 61 | + q({A{1}, A{2}, A{3}}); |
| 62 | + |
| 63 | + // mutableにするため、静的記憶域に保持される。 |
| 64 | + // 以下と等価: |
| 65 | + // const A __c[3] = {A{1}, A{2}, A{3}}; |
| 66 | + // q(std::initializer_list<A>(__c, __c+3)); |
| 67 | +} |
| 68 | +``` |
| 69 | +
|
| 70 | +将来的に、静的記憶域に配置される配列は、共有される可能性がある。この動作のために、共有で予期しない問題が起きる`mutable`は共有から除外される。 |
| 71 | +
|
| 72 | +```cpp |
| 73 | +std::initializer_list<int> i1 = {1,2,3,4,5}; |
| 74 | +std::initializer_list<int> i2 = {2,3,4}; |
| 75 | +PERMIT(i1.begin() == i2.begin() + 1); // 将来的な動作 |
| 76 | +``` |
| 77 | + |
| 78 | +## <a id="relative-page" href="#relative-page">関連項目</a> |
| 79 | +- [C++11 初期化子リスト](/lang/cpp11/initializer_lists.md) |
| 80 | +- [C++26 ファイルを読み込む`#embed`命令を追加](/lang/cpp26/embed.md) |
| 81 | + |
| 82 | + |
| 83 | +## 参照 |
| 84 | +- [P2752R3 Static storage for braced initializers](http://open-std.org/jtc1/sc22/wg21/docs/papers/2023/p2752r3.html) |
0 commit comments