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)使用分配器构造的手段在 p 所指示的未初始化内存位置,以 OUTERMOST 为分配器创建给定类型 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 具分配器,则修改 tuple xy 以包含适合的内层分配其,按照下列三条规则产生二个新的 tuple 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) ,而其构造函数接收 allocator_tag (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) ,则 xprimestd::tuple_cat(std::tuple<Args1&&...>(std::move(x)), std::tuple<inner_allocator_type&>(inner_allocator()))
应用相同规则到 T2 并以 yprime 替换 y
一旦构造了 xprimeyprime ,则在分配器的存储构造 pair 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<>()) ,即传递 pair 的成员类型上的内层分配器,若它们接受。
5) 等价于
6) 等价于
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::scoped_allocator_adaptor 作为分配器使用的具分配器对象,例如 std::vector 所调用(通过 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++17 中弃用)(C++20 中移除)
在分配的存储构造对象
(std::allocator<T> 的公开成员函数)