namespace std {
template<class T, class A>
shared_ptr<T> allocate_shared_for_overwrite(const A& a); // (1)
template<class T, class A>
shared_ptr<T> allocate_shared_for_overwrite(const A& a, size_t N); // (2)
}
概要
T
型のオブジェクト、またはその配列へのshared_ptr
を構築し、返却する。
要件
テンプレートパラメータA
はCpp17Allocator
の要件を満たす。
効果
型T
のオブジェクトにメモリを割り当てる(T
がU[]
の場合はU[N]
。N
はそれぞれのオーバーロードで指定された引数から決定される)。メモリは、引数a
のコピー(参照カウンタと型T
の本体を連続メモリ領域に配置するためにはT
のサイズより大きい領域をアロケートする必要があるためにallocator_traits<Alloc>::rebind_alloc<value_type>
を用いて再束縛されたもの)を使用して割り当てられる。
すべてのオーバーロードにおいて、確保された領域のオブジェクトはデフォルト構築される。
例外がスローされた場合、関数は効果がない。
非配列型U
の(サブ)オブジェクトがこの関数によって初期化されるときは、式::new(pv) U
によって初期化される。ここで、pv
は型 void *
を持ち、型U
のオブジェクトを保持するための適切なストレージを指す。
配列要素は、アドレスの昇順で初期化される。
戻り値によって管理されるオブジェクトのlifetime
が終了するか、配列要素の初期化が例外をスローすると、初期化された要素は元の構造の逆の順序で破棄される。
- (1) : このオーバーロードが選択されるとき、
T
は、不明な境界の配列ではない。T
型のオブジェクトへのshared_ptr
を返す。 - (2) : このオーバーロードが選択されるとき、
T
は、不明な境界の配列である。型U[N]
のオブジェクトへのshared_ptr
を返す。ここで、U
はremove_extent_t<T>
である。
戻り値
新しく構築されたオブジェクトのアドレスを格納および所有するshared_ptr
インスタンス。
事後条件
r.get() != 0 && r.use_count() == 1
, ここで、r
は戻り値である。
例外
bad_alloc
、またはallocate
またはオブジェクトの初期化からスローされた例外。
備考
同様の効果を持つ関数に、make_shared_for_overwrite()
があるが、
この関数はメモリの確保にユーザー定義のアロケータを使用したい場合などに用いることができる。
デフォルト構築においては、トリビアルにデフォルト構築可能な型のオブジェクトは未初期化状態となるため、値を読みだす前に明示的に初期化(overwrite)する必要がある。初期化後の状態が不定になってほしくない場合はこの関数ではなくallocate_shared()
を使用すべき。
例
#include <memory>
#include <iostream>
int main() {
std::allocator<int> alloc;
std::shared_ptr<int> sp = std::allocate_shared_for_overwrite<int>(alloc);
if (sp) {
*sp = 0; // 必ず初期化する
std::cout << *sp << std::endl;
}
}
出力
0
バージョン
言語
- C++20
処理系
- Clang: 10.0.0 現在未対応 ✅
- GCC: 10.0.0 現在未対応 ✅
- Visual C++: ??