- memory[meta header]
- function template[meta id-type]
- std[meta namespace]
- cpp20[meta cpp]
template<class T, class Alloc, class... Args>
T* uninitialized_construct_using_allocator(T* p, const Alloc& alloc, Args&&... args);
Alloc
型のアロケータオブジェクト alloc
、および、コンストラクタ引数 args
を用いて、p
で指定された領域に T
型のオブジェクトを uses-allocator 構築する。
以下と同等
::new(static_cast<void*>(p)) T(make_obj_using_allocator<T>(alloc, std::forward<Args>(args)...))
- make_obj_using_allocator[link make_obj_using_allocator.md]
- forward[link ../utility/forward.md]
uses_allocator_construction_args
を見ればわかる通り、uses-allocator 構築は、その名前に反して必ずしもアロケータオブジェクトを使うとは限らないので注意。
(uses_allocator_v
<T, Alloc>
がfalse
の場合、アロケータオブジェクトalloc
は無視される)
#include <iostream>
#include <vector>
#include <utility>
#include <memory>
#include <new>
// 状態付きアロケータ
template <typename T>
class MyAlloc {
public:
using value_type = T;
T* allocate(std::size_t n) { return static_cast<T*>(::operator new(sizeof(T) * n)); }
void deallocate(T* p, std::size_t n) { ::operator delete(static_cast<void*>(p), sizeof(T) * n); }
MyAlloc(int state) noexcept : state(state) {}
template <typename U>
MyAlloc(const MyAlloc<U>& o) noexcept : state(o.state) {}
int state;
};
template <typename T>
bool operator==(const MyAlloc<T>& lhs, const MyAlloc<T>& rhs) noexcept
{
return lhs.state == rhs.state;
}
template <typename T>
bool operator!=(const MyAlloc<T>& lhs, const MyAlloc<T>& rhs) noexcept
{
return lhs.state != rhs.state;
}
// 型別名
using VEC = std::vector<int, MyAlloc<int>>;
using V = std::pair<std::pair<VEC, int>, int>;
int main()
{
alignas(V) char s[sizeof(V)];
auto p = std::uninitialized_construct_using_allocator(
reinterpret_cast<V*>(s),
MyAlloc<int>{42}, std::make_pair(VEC{MyAlloc<int>(99)}, 1), 2
);
std::cout << p->first.first.get_allocator().state << '\n';
}
- uninitialized_construct_using_allocator[color ff0000]
- get_allocator[link ../vector/vector/get_allocator.md]
42
- C++20
- Clang: 9.0.0 [mark noimpl]
- GCC: 9.1.0 [mark verified]
- ICC: ??
- Visual C++: ??