命名空间
变体
操作

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)

使用 OuterAllocator 和提供的构造函数参数,在由 p 指向的已分配但未初始化的存储中构造一个对象。 如果对象是本身使用分配器的类型,或者如果是 std::pair,则将 InnerAllocator 传递给构造的对象。

首先,通过调用 this->outer_allocator() 获取最外层分配器 OUTERMOST,然后在该调用的结果上递归调用 outer_allocator() 成员函数,直到达到没有此类成员函数的分配器。

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

1) 使用 OUTERMOST 作为分配器,通过 使用分配器构造 在由 p 指示的未初始化内存位置创建一个给定类型 T 的对象。 调整后,使用 T 的构造函数所需的分配器约定,调用 OUTERMOST_ALLOC_TRAITS(*this)::construct.
此重载仅在 U 不是 std::pair 的特化时才参与重载解析。
(直到 C++20)
等同于
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)...
    )
);

(自 C++20 起)
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<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 的构造函数参数
non_pair - pair 参数,转换为 pair 以供进一步构造

[编辑] 返回值

(无)

[编辑] 注释

此函数由任何分配器感知的对象(如 std::vector)调用(通过 std::allocator_traits),该对象被赋予了 std::scoped_allocator_adaptor 作为要使用的分配器。由于 inner_allocator 本身是 std::scoped_allocator_adaptor 的实例,因此当通过此函数构造的分配器感知对象开始构造它们自己的成员时,也会调用此函数。

[编辑] 缺陷报告

以下行为更改缺陷报告被追溯应用于以前发布的 C++ 标准。

DR 应用于 已发布的行为 正确行为
LWG 2975 C++11 第一个重载在某些情况下被错误地用于对构造函数 受限于不接受对
P0475R1 C++11 对的分段构造可能会复制参数 转换为元组的引用以避免复制
LWG 3525 C++11 没有重载可以处理可转换为对的非对类型 添加了重构重载

[编辑] 另请参阅

[static]
在分配的存储空间中构造一个对象
(函数模板) [编辑]
(直到 C++20)
在分配的存储空间中构造一个对象
(std::allocator<T> 的公共成员函数) [编辑]