std::scoped_allocator_adaptor<OuterAlloc,InnerAlloc...>::construct

来自cppreference.com
 
 
动态内存管理
未初始化内存算法
受约束的未初始化内存算法
分配器
垃圾收集器支持
(C++11)(C++23 前)
(C++11)(C++23 前)
(C++11)(C++23 前)
(C++11)(C++23 前)
(C++11)(C++23 前)
(C++11)(C++23 前)



 
 
在标头 <scoped_allocator> 定义
template< class T, class... Args >
void construct( T* p, Args&&... args );
(1)
template< class T1, class T2, class... Args1, class... Args2 >

void construct( std::pair<T1, T2>* p,
                std::piecewise_construct_t,
                std::tuple<Args1...> x,

                std::tuple<Args2...> y );
(2) (C++20 前)
template< class T1, class T2 >
void construct( std::pair<T1, T2>* p );
(3) (C++20 前)
template< class T1, class T2, class U, class V >
void construct( std::pair<T1, T2>* p, U&& x, V&& y );
(4) (C++20 前)
template< class T1, class T2, class U, class V >
void construct( std::pair<T1, T2>* p, const std::pair<U, V>& xy );
(5) (C++20 前)
template< class T1, class T2, class U, class V >
void construct( std::pair<T1, T2>* p, std::pair<U, V>&& xy );
(6) (C++20 前)
template< class T1, class T2, class NonPair >
void construct( std::pair<T1, T2>* p, NonPair&& non_pair );
(7) (C++20 前)

p 所指向的分配但未初始化的存储中,用 OuterAllocator 与提供的构造函数实参构造对象。若对象是自身使用分配器的类型,或若它是 std::pair,则下传 InnerAllocator 给构造的对象。

首先,获得最外层分配器 OUTERMOST:调用 this->outer_allocator(),然后在此调用的结果上递归调用 outer_allocator() 成员函数,直至抵达无这种成员函数的分配器。

定义 OUTERMOST_ALLOC_TRAITS(x)std::allocator_traits<std::remove_reference_t<decltype(OUTERMOST(x))>>

1)使用分配器构造的手段在 p 所指示的未初始化内存位置,以 OUTERMOST 为分配器创建给定类型 T 的对象。在为 T 的构造函数所期待的使用分配器约定调整后,调用 OUTERMOST_ALLOC_TRAITS(*this)::construct
此重载只有在 U 不是 std::pair 的特化时才会参与重载决议。
等价于
std::apply(

    [p, this](auto&&... newargs)
    {
        OUTERMOST_ALLOC_TRAITS(*this)::construct(
            OUTERMOST(*this), p, std::forward<decltype(newargs)>(newargs)...);
    },
    std::uses_allocator_construction_args(
        inner_allocator(),
        std::forward<Args>(args)...
    )
);

2) 首先,若 T1T2 具分配器,则按照下列三条规则修改元组 xy 以包含适合的内层分配器,产生两个新的元组 xprimeyprime
2a)T1 不具分配器
(std::uses_allocator<T1, inner_allocator_type>::value == false),则 xprimestd::tuple<Args1&&...>(std::move(x))。(亦要求 std::is_constructible<T1, Args1...>::value == true)。
2b)T1 具分配器 (std::uses_allocator<T1, inner_allocator_type>::value == true),而其构造函数接受分配器标签

std::is_constructible<T1, std::allocator_arg_t,
                      inner_allocator_type&, Args1...>::value == true

xprime

std::tuple_cat(std::tuple<std::allocator_arg_t, inner_allocator_type&>(
                    std::allocator_arg, inner_allocator()
               ),
               std::tuple<Args1&&...>(std::move(x)))
2c)T1 具分配器 (std::uses_allocator<T1, inner_allocator_type>::value == true),且其构造函数接收分配器为末尾实参

std::is_constructible<T1, Args1..., inner_allocator_type&>::value == true
xprime

std::tuple_cat(std::tuple<Args1&&...>(std::move(x)),
               std::tuple<inner_allocator_type&>(inner_allocator()))
应用相同规则到 T2 并以 yprime 替换 y
一旦构造了 xprimeyprime,则在所分配的存储中构造对偶 p,通过调用
std::allocator_traits<O>::construct(OUTERMOST,
                                    p,
                                    std::piecewise_construct,
                                    std::move(xprime),
                                    std::move(yprime));
3) 等价于

construct(p, std::piecewise_construct, std::tuple<>(), std::tuple<>())

即若它们接受,则将内层分配器传递给对偶的各成员类型。
4) 等价于
construct(p, std::piecewise_construct, std::forward_as_tuple(std::forward<U>(x)),
                                       std::forward_as_tuple(std::forward<V>(y)))
5) 等价于
construct(p, std::piecewise_construct, std::forward_as_tuple(xy.first),
                                       std::forward_as_tuple(xy.second))
6) 等价于
construct(p, std::piecewise_construct,
             std::forward_as_tuple(std::forward<U>(xy.first)),
             std::forward_as_tuple(std::forward<V>(xy.second)))
7) 此重载只有在给定仅用于阐释的函数模板

template<class A, class B>
void /*deduce-as-pair*/(const std::pair<A, B>&);

/*deduce-as-pair*/(non_pair) 在当作不求值操作数时非良构时才会参与重载决议。

等价于 construct<T1, T2, T1, T2>(p, std::forward<NonPair>(non_pair));
(C++20 前)

参数

p - 指向被分配但未初始化的存储的指针
args... - 传递给 T 的构造函数的构造函数实参
x - 传递给 T1 的构造函数的构造函数实参
y - 传递给 T2 的构造函数的构造函数实参
xy - 两个成员分别为 T1T2 构造函数实参的 pair
non_pair - 转换成 pair 以进行进一步构造的非 pair 实参

返回值

(无)

注解

此函数由任何具分配器对象,例如 std::vector,在给予了 std::scoped_allocator_adaptor 作为所用分配器时(通过 std::allocator_traits)调用。因为 inner_allocator 自身是 std::scoped_allocator_adaptor 的实例,此函数亦在具分配器对象通过此函数开始构造其自身成员时得到调用。

缺陷报告

下列更改行为的缺陷报告追溯地应用于以前出版的 C++ 标准。

缺陷报告 应用于 出版时的行为 正确行为
LWG 2975 C++11 某些情况下 pair 构造错误地使用第一重载 制约为不接受 pair
P0475R1 C++11 pair 逐片构造可能复制参数 变换到引用的 tuple 以避免复制
LWG 3525 C++11 没有能处理可转换到 pair 的非 pair 类型 添加了重构的重载

参阅

[静态]
在已分配存储中构造对象
(函数模板)
(C++20 前)
在分配的存储中构造对象
(std::allocator<T> 的公开成员函数)