空の基底の最適化
空の基底の部分オブジェクトのサイズをゼロにすることを可能とします。
[編集] 説明
あらゆるオブジェクトまたはメンバ部分オブジェクトのサイズは、たとえその型が空のクラス型 (つまり非静的データメンバを持たないクラスまたは構造体) であっても、 ([[no_unique_address]] でない限り -- 下記参照) (C++20以上) 同じ型の異なるオブジェクトのアドレスが必ず異なることを保証することが可能であるように、少なくとも1であることが要求されます。
しかし、基底クラス部分オブジェクトにはそのような制約がなく、オブジェクトのレイアウトから完全に削除する最適化が行えます。
空の基底クラスのいずれかが最初の非静的データメンバの型またはその基底でもある場合は、同じ型の2つの基底部分オブジェクトは最も派生した型のオブジェクト表現内で異なるアドレスを持つことが要求されるため、空の基底の最適化は禁止されます。
そのような状況の一般的な例は std::reverse_iterator のナイーブな実装です (空の基底 std::iterator から派生します)。 これはベースとなるイテレータ (これも std::iterator から派生します) をその最初の非静的データメンバとして保持します。
#include <cassert> struct Base {}; // 空のクラス。 struct Derived1 : Base { int i; }; struct Derived2 : Base { Base c; // Base (1バイトを占めます) の後に i のためのパディングがあります。 int i; }; struct Derived3 : Base { Derived1 c; // Base から派生。 sizeof(int) バイトを占めます。 int i; }; int main() { // 空の基底の最適化は適用されません。 // 基底が1バイトを占め、 Base メンバが1バイトを占め、 // int のアライメント要件を満たすために2バイトのパディングが続きます。 assert(sizeof(Derived2) == 2*sizeof(int)); // 空の基底の最適化は適用されません。 // 基底が少なくとも1バイトに加えて // 最初のメンバ (int と同じアライメント) のアライメント要件を // 満たすためのパディングを占めます。 assert(sizeof(Derived3) == 3*sizeof(int)); }
reinterpret_cast を用いて変換された標準レイアウトオブジェクトへのポインタがその最初のメンバを指すという要件を維持するために、空の基底の最適化は StandardLayoutType に対しては必須です。 これは標準レイアウト型に対する要件に「すべての非静的データメンバが同じクラス内で宣言されている (派生クラスですべてを宣言するか、いずれかの基底クラスですべてを宣言するか、のいずれか)」および「最初の非性的データメンバとして同じ型の基底クラスを持たない」がある理由です。 |
(C++11以上) |
空のメンバ部分オブジェクトは、 |
(C++20以上) |
[編集] ノート
空の基底の最適化は、アロケータがステートレスな場合にアロケータメンバのための追加の記憶域を占めることを避けるために、アロケータ対応の標準ライブラリクラス (std::vector、 std::function、 std::shared_ptr など) で一般的に使用されています。 これは要求されるデータメンバ (begin や end、 vector の場合は capacity ポインタなど) のいずれかをアロケータと共に boost::compressed_pair の同等品に格納することによって達成されます。
[編集] 参考文献
- C++11 standard (ISO/IEC 14882:2011):
- 5.10 Equality operators [expr.eq](p: 2)
- 5.3.3 Sizeof [expr.sizeof](p: 2)
- 9 Classes [class](p: 4,7)
- 9.2 Class members [class.mem](p: 20)
- C++98 standard (ISO/IEC 14882:1998):
- 5.10 Equality operators [expr.eq](p: 2)
- 5.3.3 Sizeof [expr.sizeof](p: 2)
- 9 Classes [class](p: 3)