C++标准模板(STL)- 类型支持 (杂项变换, 确定类型组的共用引用类型,std::common_reference)

类型特性

类型特性定义一个编译时基于模板的结构,以查询或修改类型的属性。

试图特化定义于 <type_traits> 头文件的模板导致未定义行为,除了 std::common_type 可依照其所描述特化。

定义于<type_traits>头文件的模板可以用不完整类型实例化,除非另外有指定,尽管通常禁止以不完整类型实例化标准库模板。

杂项变换

确定类型组的共用引用类型

复制代码
std::common_reference

|---------------------------------------------------|---|-----------|
| 定义于头文件 <type_traits> | | |
| template< class... T > struct common_reference; | | (C++20 起) |

确定类型 T... 的共用引用类型,即 T... 中所有类型能转换或绑定到的类型。若这种类型存在(如按照后述规则所确定),则成员 type 指名该类型。否则,无成员 type 。若 T... 中的任何类型为(可有 cv 限定的) void 以外的不完整类型,则行为未定义。

给定引用类型是, common_reference 试图寻找所有提供的引用类型都能绑定到的引用类型,但若找不到这种引用类型,则可能返回非引用类型。

  • 若 sizeof...(T) 为零,则无成员 type
  • 若 sizeof...(T) 为一(即 T... 仅含一个类型 T0 ),则成员 type 指名与 T0 相同的类型。
  • 若 sizeof...(T) 为二(即 T... 含二个类型 T1T2 ),则:
    • T1T2 都是引用类型,而 T1T2简单共用引用类型 S 存在,则成员类型 type 指名 S
    • 否则,若 std::basic_common_reference<std::remove_cvref_t<T1>, std::remove_cvref_t<T2>, T1Q, T2Q>::type 存在,其中 TiQ 是一元别名模板,满足 TiQ<U> 为 U 带上 Ti 的 cv 及引用限定符,则成员类型 type 指名该类型;
    • 否则,若 decltype(false? val<T1>() : val<T2>()) 是合法类型,其中 val 为函数模板 template<class T> T val(); ,则成员类型 type 指名该类型;
    • 否则,若 std::common_type_t<T1, T2> 为合法类型,则成员类型 type 代表该类型;
    • 否则,无成员 type
  • 若 sizeof...(T) 大于二(即 T...T1, T2, R... 组成),则若 std::common_reference_t<T1, T2> 存在,且 std::common_reference_t<std::common_reference_t<T1, T2>, R...> 存在,则成员类型 type 指代它。所有其他情况下,无成员类型 type

二个引用类型 T1T2简单共用引用类型定义如下:

  • T1cv1 X &T2cv2 Y & (即都是左值引用):则其简单共用引用类型为 decltype(false? std::declval<cv12 X &>() : std::declval<cv12 Y &>()) ,其中 cv12cv1cv2 的联合,若该类型存在且为引用类型;
  • T1 a与 T2 均为右值引用类型:若 T1 &T2 & 的简单共用引用类型(按照前一条确定)存在,则令 C 代表该类型的对应右值引用类型。若 std::is_convertible_v<T1, C> 与 std::is_convertible_v<T2, C> 均为 true ,则 T1T2 的简单共用引用类型为 C
  • 否则,二个类型之一必须为左值引用类型 A & 而另一个必须为右值引用类型 B &&AB 可为 cv 限定)。令 D 代表 A & 与 B const & 的简单共用引用类型,若它存在。若 D 存在且 std::is_convertible_v<B &&, D> 为 true ,则简单共用引用类型为 D
  • 否则,无简单共用引用类型。

成员类型

|--------|-------------------|
| 名称 | 定义 |
| type | 所有 T... 的共用引用类型 |

辅助类型

|---------------------------------------------------------------------------------------------------------------------------------|---|---|
| template< class... T > using common_reference_t = typename std::common_reference<T...>::type; | | |
| template< class T, class U, template<class> class TQual, template<class> class UQual > struct basic_common_reference { }; | | |

类模板 basic_common_reference 是定制点,允许用户影响 common_reference 对用户定义类型(常为代理引用)的结果。初等模板为空。

特化

若 std::is_same<T, std::decay_t<T>> 与 std::is_same<U, std::decay_t<U>> 皆为 true ,且它们至少有一个依赖于程序定义类型,则程序可以在前二个形参 TU 上特化 basic_common_reference<T, U, TQual, UQual>

若这种特化拥有名为 type 的成员,则它必须是指名 TQual<T> 和 UQual<U> 都能转换到的类型的公开且无歧义的成员类型。另外, std::basic_common_reference<T, U, TQual, UQual>::type 和 std::basic_common_reference<U, T, UQual, TQual>::type 必须代表同一类型。

程序不可在第三或第四形参上特化 basic_common_reference ,亦不可特化 common_reference 自身。以违背这些规则的方式添加特化的程序拥有未定义行为。

调用示例

复制代码
#include <type_traits>
#include <typeinfo>
#include <iostream>

namespace std
{
template <class... Ts>
struct common_reference;

template <class T, class U>
struct common_reference<T, U>
{
    using type = decltype(std::declval<T>() = std::declval<U>());
};

template <class T, class U, class... Args>
struct common_reference<T, U, Args...>
{
    using type = common_reference<typename common_reference<T, U>::type, Args...>;
};

template <class T, class U>
using common_reference_t = typename common_reference<T, U>::type;
} // namespace std

int main()
{
    std::cout << "std::common_reference < int&, int&& >         :     "
              <<  typeid(std::common_reference < int&, int&& >).name() << std::endl;
    std::cout << "std::common_reference_t < int&, int&& >       :     "
              <<  typeid(std::common_reference_t < int&, int&& >).name() << std::endl;
    std::cout << "-----------------------------------------------" << std::endl;
    std::cout << std::endl;

    std::cout << "std::common_reference<int&, double&>          :     "
              <<  typeid(std::common_reference<int&, double&>).name() << std::endl;
    std::cout << "std::common_reference_t<int&, double&>        :     "
              <<  typeid(std::common_reference_t<int&, double&>).name() << std::endl;
    std::cout << "-----------------------------------------------" << std::endl;
    std::cout << std::endl;

    std::cout << "std::common_reference<double&, double&&>      :     "
              <<  typeid(std::common_reference < double &, double && >).name() << std::endl;
    std::cout << "std::common_reference_t<double&, double&&>    :     "
              <<  typeid(std::common_reference_t < double&, double && >).name() << std::endl;
    std::cout << "-----------------------------------------------" << std::endl;
    std::cout << std::endl;

    std::cout << "std::common_reference<bool&, bool&&>          :     "
              <<  typeid(std::common_reference < bool &, bool && >).name() << std::endl;
    std::cout << "std::common_reference_t<bool&, bool&&>        :     "
              <<  typeid(std::common_reference_t < bool&, bool && >).name() << std::endl;
    std::cout << "-----------------------------------------------" << std::endl;
    std::cout << std::endl;

    std::cout << "std::common_reference<bool&, const bool&&>    :     "
              <<  typeid(std::common_reference < bool &, const bool && >).name() << std::endl;
    std::cout << "std::common_reference_t<bool&, const bool&&>  :     "
              <<  typeid(std::common_reference_t < bool&, const bool && >).name() << std::endl;
    std::cout << "-----------------------------------------------" << std::endl;
    std::cout << std::endl;

    std::cout << "std::common_reference<char&, char&&>          :     "
              <<  typeid(std::common_reference < char &, char && >).name() << std::endl;
    std::cout << "std::common_reference_t<char&, char&&>        :     "
              <<  typeid(std::common_reference_t < char&, char && >).name() << std::endl;
    std::cout << "-----------------------------------------------" << std::endl;
    std::cout << std::endl;

    return 0;
}

输出

复制代码
std::common_reference < int&, int&& >         :     St16common_referenceIJRiOiEE
std::common_reference_t < int&, int&& >       :     i
-----------------------------------------------

std::common_reference<int&, double&>          :     St16common_referenceIJRiRdEE
std::common_reference_t<int&, double&>        :     i
-----------------------------------------------

std::common_reference<double&, double&&>      :     St16common_referenceIJRdOdEE
std::common_reference_t<double&, double&&>    :     d
-----------------------------------------------

std::common_reference<bool&, bool&&>          :     St16common_referenceIJRbObEE
std::common_reference_t<bool&, bool&&>        :     b
-----------------------------------------------

std::common_reference<bool&, const bool&&>    :     St16common_referenceIJRbOKbEE
std::common_reference_t<bool&, const bool&&>  :     b
-----------------------------------------------

std::common_reference<char&, char&&>          :     St16common_referenceIJRcOcEE
std::common_reference_t<char&, char&&>        :     c
-----------------------------------------------
相关推荐
_r0bin_3 小时前
前端面试准备-7
开发语言·前端·javascript·fetch·跨域·class
zhang98800003 小时前
JavaScript 核心原理深度解析-不停留于表面的VUE等的使用!
开发语言·javascript·vue.js
Fanxt_Ja5 小时前
【JVM】三色标记法原理
java·开发语言·jvm·算法
蓝婷儿5 小时前
6个月Python学习计划 Day 15 - 函数式编程、高阶函数、生成器/迭代器
开发语言·python·学习
love530love5 小时前
【笔记】在 MSYS2(MINGW64)中正确安装 Rust
运维·开发语言·人工智能·windows·笔记·python·rust
南郁6 小时前
007-nlohmann/json 项目应用-C++开源库108杰
c++·开源·json·nlohmann·现代c++·d2school·108杰
slandarer6 小时前
MATLAB | 绘图复刻(十九)| 轻松拿捏 Nature Communications 绘图
开发语言·matlab
狐凄6 小时前
Python实例题:Python计算二元二次方程组
开发语言·python
roman_日积跬步-终至千里7 小时前
【Go语言基础【3】】变量、常量、值类型与引用类型
开发语言·算法·golang
roman_日积跬步-终至千里7 小时前
【Go语言基础】基本语法
开发语言·golang·xcode