std::unwrap_reference, std::unwrap_ref_decay

来自cppreference.com
< cpp‎ | utility‎ | functional
 
 
工具库
语言支持
类型支持(基本类型、RTTI)
库功能特性测试宏 (C++20)
动态内存管理
程序工具
协程支持 (C++20)
变参数函数
调试支持
(C++26)
三路比较
(C++20)
(C++20)(C++20)(C++20)
(C++20)(C++20)(C++20)
通用工具
日期和时间
函数对象
格式化库 (C++20)
(C++11)
关系运算符 (C++20 中弃用)
整数比较函数
(C++20)(C++20)(C++20)   
(C++20)
交换类型运算
(C++14)
(C++11)
(C++11)
(C++11)
(C++17)
常用词汇类型
(C++11)
(C++17)
(C++17)
(C++17)
(C++11)
(C++17)
(C++23)
初等字符串转换
(C++17)
(C++17)

 
函数对象
函数调用
(C++17)(C++23)
恒等函数对象
(C++20)
引用包装器
(C++11)(C++11)
unwrap_referenceunwrap_ref_decay
(C++20)(C++20)
通透运算符包装器
(C++14)
(C++14)
(C++14)
(C++14)  
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)
(C++14)

旧式绑定器与适配器
(C++17 前*)
(C++17 前*)
(C++17 前*)
(C++17 前*)
(C++17 前*)(C++17 前*)(C++17 前*)(C++17 前*)
(C++20 前*)
(C++20 前*)
(C++17 前*)(C++17 前*)
(C++17 前*)(C++17 前*)

(C++17 前*)
(C++17 前*)(C++17 前*)(C++17 前*)(C++17 前*)
(C++20 前*)
(C++20 前*)
 
在标头 <type_traits> 定义
在标头 <functional> 定义
template< class T >
struct unwrap_reference;
(1) (C++20 起)
template< class T >
struct unwrap_ref_decay;
(2) (C++20 起)
1)T 为某类型 Ustd::reference_wrapper<U>,则提供指名 U& 的成员类型别名 type;否则提供指名 T 的成员类型别名 type
2)T 为某类型 Ustd::reference_wrapper<U>,忽略 cv 限定和引用性,则提供指名 U& 的成员类型别名 type;否则提供指名 std::decay_t<T> 的成员类型别名 type

如果程序添加了此页面上描述的任何模板的特化,那么行为未定义。

成员类型

名称 定义
type

(1)Tstd::reference_wrapper<U> 则为 U&;否则为 T
(2)std::decay_t<T>std::reference_wrapper<U> 则为 U&;否则为 std::decay_t<T>

辅助类型

template<class T>
using unwrap_reference_t = unwrap_reference<T>::type;
(1) (C++20 起)
template<class T>
using unwrap_ref_decay_t = unwrap_ref_decay<T>::type;
(2) (C++20 起)

可能的实现

template<class T>
struct unwrap_reference { using type = T; };
template<class U>
struct unwrap_reference<std::reference_wrapper<U>> { using type = U&; };
 
template<class T>
struct unwrap_ref_decay : std::unwrap_reference<std::decay_t<T>> {};

注解

std::unwrap_ref_decay 进行与 std::make_pairstd::make_tuple 所用者相同的变换。

功能特性测试 标准 功能特性
__cpp_lib_unwrap_ref 201811L (C++20) std::unwrap_ref_decaystd::unwrap_reference

示例

#include <cassert>
#include <functional>
#include <iostream>
#include <type_traits>
 
int main()
{
    static_assert(std::is_same_v<std::unwrap_reference_t<int>, int>);
    static_assert(std::is_same_v<std::unwrap_reference_t<const int>, const int>);
    static_assert(std::is_same_v<std::unwrap_reference_t<int&>, int&>);
    static_assert(std::is_same_v<std::unwrap_reference_t<int&&>, int&&>);
    static_assert(std::is_same_v<std::unwrap_reference_t<int*>, int*>);
 
    {
        using T = std::reference_wrapper<int>;
        using X = std::unwrap_reference_t<T>;
        static_assert(std::is_same_v<X, int&>);
    }
    {
        using T = std::reference_wrapper<int&>;
        using X = std::unwrap_reference_t<T>;
        static_assert(std::is_same_v<X, int&>);
    }
 
    static_assert(std::is_same_v<std::unwrap_ref_decay_t<int>, int>);
    static_assert(std::is_same_v<std::unwrap_ref_decay_t<const int>, int>);
    static_assert(std::is_same_v<std::unwrap_ref_decay_t<const int&>, int>);
 
    {
        using T = std::reference_wrapper<int&&>;
        using X = std::unwrap_ref_decay_t<T>;
        static_assert(std::is_same_v<X, int&>);
    }
 
    {
        auto reset = []<typename T>(T&& z)
        {
        //  x = 0; // 错误: 如果 T 为 reference_wrapper<> 则无法工作
            // 对于普通类型将 T&& 转换为 T&
            // 对于 reference_wrapper<U> 将 T&& 转换为 U&
            decltype(auto) r = std::unwrap_reference_t<T>(z);
            std::cout << "r: " << r << '\n';
            r = 0; // OK, r 具有引用类型
        };
 
        int x = 1;
        reset(x);
        assert(x == 0);
 
        int y = 2;
        reset(std::ref(y));
        assert(y == 0);
    }
}

输出:

r: 1
r: 2

参阅

可复制构造 (CopyConstructible) 可复制赋值 (CopyAssignable) 的引用包装器
(类模板)
创建一个 pair 对象,其类型根据各实参类型定义
(函数模板)
创建一个 tuple 对象,其类型根据各实参类型定义
(函数模板)