std::make_from_tuple

来自cppreference.com
< cpp‎ | utility
 
 
工具库
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)
(C++20)
swap 与类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
make_from_tuple
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)
 
在标头 <tuple> 定义
template< class T, class Tuple >
constexpr T make_from_tuple( Tuple&& t );
(C++17 起)

构造 T 类型对象,以元组 t 的元素为构造函数的参数。

若假设的变量定义 T var(std::get<Is>(std::forward<Tuple>(t))...); 非良构,或 T 为引用类型且 var绑定到临时对象 (C++23 起),其中 Is... 代表形参包 0, 1, ..., std::tuple_size_v<std::remove_reference_t<Tuple>> - 1 ,则程序非良构。

参数

t - 元组,其元素被用作 T 构造函数的参数

返回值

构造的 T 对象或引用。

注解

元组不必是 std::tuple ,可以为任何支持 std::getstd::tuple_size 的类型所替代;特别是可以用 std::arraystd::pair

因为受保证的复制消除,不需要 T 为可移动。

可能的实现

namespace detail {
template<class T, class Tuple, std::size_t... I>
constexpr T make_from_tuple_impl(Tuple&& t, std::index_sequence<I...>)
{
    static_assert(std::is_constructible_v<T,
        decltype(std::get<I>(std::declval<Tuple>()))...>);
#if __cpp_lib_reference_from_temporary >= 202202L
    if constexpr (std::tuple_size_v<std::remove_reference_t<Tuple>> == 1) {
        using tuple_first_t = decltype(std::get<0>(std::declval<Tuple>()));
        static_assert(!std::reference_constructs_from_temporary_v<T, tuple_first_t>);
    }
#endif
    return T(std::get<I>(std::forward<Tuple>(t))...);
}
} // namespace detail
 
template<class T, class Tuple>
constexpr T make_from_tuple(Tuple&& t)
{
    return detail::make_from_tuple_impl<T>(std::forward<Tuple>(t),
        std::make_index_sequence<std::tuple_size_v<std::remove_reference_t<Tuple>>>{});
}

示例

#include <iostream>
#include <tuple>
 
struct Foo {
    Foo(int first, float second, int third) {
        std::cout << first << ", " << second << ", " << third << "\n";
    }
};
 
int main()
{
   auto tuple = std::make_tuple(42, 3.14f, 0);
   std::make_from_tuple<Foo>(std::move(tuple));
}

输出:

42, 3.14, 0

缺陷报告

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

缺陷报告 应用于 出版时的行为 正确行为
LWG 3528 C++17 在 1-元组的情况下允许含有 reinterpret_cast 等的转型 不允许

参阅

创建一个 tuple 对象,其类型根据各实参类型定义
(函数模板)
创建转发引用tuple
(函数模板)
(C++17)
以一个实参的元组来调用函数
(函数模板)