C++17之std::void_t

目录

1.std::void_t 的原理

2.std::void_t 的应用

2.1.判断成员存在性

2.1.1.判断嵌套类型定义

2.1.2 判断成员是否存在

2.2 判断表达式是否合法

2.2.1 判断是否支持前置++运算符

2.2.3 判断两个类型是否可做加法运算

3.std::void_t 与 std::enable_if


1.std::void_t 的原理

        std::void_t<> 是 C++ 17 标准增加的一个非常实用的功能,其实就是一个模板别名的定义,它的作用是将一系列不同的类型映射成 void 类型。在 C++ 20 的概念(concept)推出之前,它被认为是使用 SFINAE 机制的最方便的方法之一。std::void_t<> 根据对一个表达式(比如用 decltype 推断一个操作数的类型的表达式)的有效性判断,从候选集中移除某些符合条件的模板函数或模板类,从而允许特定的函数重载或类模板的特化版本成立。

        标准库中这样定义 std::void_t:

template <class... _Types>
using void_t = void;

        std::void_t<> 定义了一个模板别名,它的作用是将模板参数列表中的 _Types 映射成 void 类型。这个别名能工作的前提是 _Types 中的每种类型都必须是合法的类型,若 _Types 中包含不合法的类型,则这个别名的定义非法,从而触发 SFINAE 机制产生相应的效果。std::void_t<> 的灵活性在于 _Types  甚至可以是 decltype() 表达式,比如 std::void_t<decltype(5)> 。这个表达式不会触发模板参数替换失败,因为 decltype(5) 推断得到 int 类型,整个表达式最终的结果相当于 void。但是 std::void_t<decltype(std::string::to_integer)> 就会失败,因为目前标准库的 std::string 没有名为 to_integer 的成员,decltype 推断不出一个合法的类型。这就是 std::void_t<>的工作原理,当 std::void_t<> 替换失败时,就会触发 SFINAE 机制删除相应的模板参数替换得到的错误结果。

2.std::void_t 的应用

2.1.判断成员存在性

2.1.1.判断嵌套类型定义

        std::void_t<> 可以用来判断一个类型 T 是否嵌套定义了某个类型。判断的原理就是利用 T::SomeType 类型的存在性,转化成 std::void_t<T::SomeType> 定义的合法性。来看下面的例子,判断某个类型是否内部嵌套定义了子类型:InnerType:

template< class, class = std::void_t<> >  //或者 template< class, class = void >
struct has_type_member : std::false_type { };template< class T >
struct has_type_member<T, std::void_t<typename T::InnerType>> : std::true_type { };// 演示代码
struct ObjectA {enum class InnerType { COLOR, SHAPE };
};
struct ObjectB {using InnerType = int;
};
struct ObjectC {
};static_assert(has_type_member<ObjectA>::value); //true, ObjectA 有 enum 类型的 InnerType 定义
static_assert(has_type_member<ObjectB>::value); //true, ObjectB 有 int 类型别名 InnerType
static_assert(has_type_member<ObjectC>::value); //false,ObjectC 没有 InnerType

        这里说明一下,因为 `std::void_t<>` 定义中的`_Types`是一个变长参数列表,可以是空,所以当 `_Types` 为空的时候也是一个合法的模板别名,即 `std::void_t<> == void`。上述代码中的 `class = std::void_t<>` 很多人也直接写成 `class = void`,效果是一样的。模板实例化过程中的匹配顺序,`void` 总是优先级最低的,所以编译器总是优先匹配第二个 `has_type_member<>` 定义(`has_type_member<>`的偏特化版本),得到`std::true_type<>`定义的 `value`。只有当 `T::InnerType` 不存在,导致`std::void_t<>` 定义失败的时候,编译器才会继续匹配第一个 `has_type_member<>` 的泛化版本。`has_type_member<>` 的泛化版本虽然因为 `void` 优先级低,总是最后才轮到,但是它能匹配任意情况,得到`std::false_type<>`定义的 `value`。根据 SFINAE 机制,编译器对之前的失败也不报错,于是 `has_type_member<>` 就完美地实现了自己的设计意图。

        `std::false_type<>` 和 `std::true_type<>` 是 `std::integral_constant<>` 的一个特化版本,`has_type_member<>`借助这个继承关系得到了一个类型为 bool 的静态变量 value。

2.1.2 判断成员是否存在

        借助于上一节的思路,判断一个类型中是否存在某个数据成员的方法非常简单,但是需要注意的是,因为`MemberName` 是数据成员的名字,所以`T::MemberName`就不是一个类型了,不能用做`std::void_t<>`的模板参数。这时候就要用到 `decltype()`了,让编译器推导出数据成员的类型:

template< class, class = void >
struct has_data_member : std::false_type { };template< class T >
struct has_data_member<T, std::void_t<decltype(T::Tom)>> : std::true_type { };

        但是对于成员函数的判断,就不能直接使用 `decltype(T::Func)`,因为这只适用于静态成员函数的情况,对于非静态的成员函数来说,T::Func 是一个非法的表达式。假设 a 是 T 类型的一个对象实例,则 `a.T::Func` 或 `a.func` 才是合法的表达式。所以对于成员函数来说,如果还是使用 `decltype(T::Func)`的语法形式,则无论的一项是否有名为 Func 的成员函数,都会因为表达式非法而替换失败,从而匹配成 false_type 的结果。难道还要构造一个 T 类型的对象才行吗?当然不是,C++ 不是还有`std::declval()`嘛。

        `std::declval() `的作用是返回任意类型 T 的右值引用类型 T&& ,可以借助这个右值引用调用 T 的成员函数,最终的效果就是在没有构造 T 的任何实例的情况下调用了 T 的成员函数,当然,这一切都是在编译期间完成的,编译器甚至都不需要函数的完整定义。具体的做法就是用`std::declval() `得到对象的右值引用,然后使用这个右值引用调用成员函数,再用`decltype()`推导函数调用返回值的类型:

std::void_t<decltype(std::declval<T>().Func())>

        如果 T 中存在名为 `Func`的成员函数,则`std::declval<T>().Func()`就是一个合法的函数调用表达式,`decltype()`就能推导出函数返回值的类型,`std::void_t<>`的模板参数就是一个合法的类型,于是它的别名定义就合法。如果 T 中不存在名为 `Func`的成员函数,则`std::declval<T>().Func()`不是一个合法的表达式,最终`std::void_t<>`的定义就是错误的,这会触发 SFINAE 机制删除错误的替换结果,从而达到选择的目的。

        根据上述分析,判断类型是否存在指定成员函数的实现可以这样做:

template<class T, class = std::void_t<>>
struct has_func_member : std::false_type {};template<class T>
struct has_func_member<T, std::void_t<decltype(std::declval<T>().fun())>> : std::true_type {};struct ObjectA {int fun() { return 42; }
};struct ObjectB {
};static_assert(has_func_member<ObjectA>::value);
static_assert(!has_func_member<ObjectB>::value);

2.2 判断表达式是否合法

        `std::void_t<>`不仅可用于判断成员的存在性,还可用来判断表达式是否合法,实际上,2.1.2 节介绍成员函数的存在性判断时,就是利用了表达式是否合法的方式处理的。这一节,我们再介绍几个这样的例子。

2.2.1 判断是否支持前置++运算符

        是的,实现思路也是尝试调用对象的前置 ++ 运算符,如果调用失败说明对象不支持前置 ++ 运算符:

template< class, class = void >
struct has_pre_increment_member : std::false_type { };template< class T >
struct has_pre_increment_member<T, std::void_t< decltype(++std::declval<T&>()) > > : std::true_type { };

        `std::declval<T&>()`得到一个右值引用,对这个右值引用调用前置 ++ 运算符,如果表达式合法,则`std::void_t<decltype(++std::declval<T&>())>`的定义就是合法的,`has_pre_increment_member<>` 的 true_type 特化版本实例化成为最佳匹配。反之,若对前置 ++ 运算符的调用失败,则`has_pre_increment_member<>`特化版本就得到一个错误的替换结果,编译器于是“不动声色地”按照`has_pre_increment_member<>`的泛化版本实例化出 false_type 的版本。

2.2.2 判断是否支持迭代器

        C++ 标准库中的容器都支持通过 begin() 和 end() 函数获得对应的迭代器,可以借助对这两个成员函数的存在性判断一个类型是否支持迭代器,当然,这不一定严谨,我们只是用这个例子展示 `std::void_t<>`的更多用法。

template <typename, typename = void>
struct is_iterable : std::false_type {};template <typename T>
struct is_iterable<T, std::void_t< decltype(std::declval<T>().begin()), decltype(std::declval<T>().end()) > > : std::true_type {};

对了,谁说 `std::void_t<>`只能用一个模板参数,这个例子不就用了两个嘛。

2.2.3 判断两个类型是否可做加法运算

        实现的思路仍然是使用`std::declval()`得到两个对象的右值引用,然后让它们“加”一下,看看“加”的表达式是否合法。

template<typename U, typename V, typename T = void> 
struct is_can_add : std::false_type  { };template<typename U, typename V>
struct is_can_add<U, V, std::void_t< decltype(std::declval<U>() + std::declval<V>()) > > : std::true_type { };

3.std::void_t 与 std::enable_if

        `std::enable_if<>` 可以用在函数返回值上,也可以用在函数参数或模板参数上,它的原理就是借助 `std::enable_if<>` 的失效条件产生错误的函数签名,然后利用 SFINAE 机制从函数候选集中移除替换失败的函数,从而达到选择特定的函数重载形式或类模板的实例化结果的目的,这是`std::enable_if<>`的使用特点。

        `std::void_t<>`则可以用来判断一个类型的正确性或存在性,借助于`decltype` 和`std::declval()`,还可以用来判断一个表达式是否合法。如果存在性和合法性判断失败将导致`std::void_t<>`的定义失败,利用这一点配合 SFINAE 机制也可以实现在编译其的一些类型选择。

        与传统的使用 SFINAE 机制的方法相比,`std::enable_if<>`和`std::void_t<>`具有更简洁的语义表达方式和更直观的语法形式,使用的方法也很方便。它们一直被认为是 C++ 20 的概念(concept)推出之前使用 SFINAE 机制的最佳方式。

推荐阅读

C++反射之检测struct或class是否实现指定函数

C++之std::declval

C++之std::enable_if

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/339056.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

相机等效焦距

1. 背景 物理焦距我们很熟悉,但是在接触实际的相机参数时,相机厂家会提到一个参数等效焦距,甚至有时候不提供物理焦距,这时候如果我们得到真实的物理焦距需要进行一定的转换.在介绍两者之间的转换关系前,先介绍一下等效焦距的由来. 如上图,假设在某一个镜头,其成像面会出现图…

操作系统 - 文件管理

文件管理 考纲内容 文件 文件的基本概念&#xff1b;文件元数据和索引节点(inode) 文件的操作&#xff1a;建立&#xff0c;删除&#xff0c;打开&#xff0c;关闭&#xff0c;读&#xff0c;写 文件的保护&#xff1b;文件的逻辑结构&#xff1b;文件的物理结构目录 目录的基…

Multipass虚拟机磁盘扩容

Multipass 是一个用于轻松创建和管理 Ubuntu 虚拟机的工具&#xff0c;特别适合开发环境。要使用 Multipass 扩大虚拟机的磁盘容量&#xff0c;你需要经历几个步骤&#xff0c;因为 Multipass 自身并不直接提供图形界面来调整磁盘大小。不过&#xff0c;你可以通过结合 Multipa…

UE5 Http Server

前言 最近要用UE 作为一个服务器去接收来自外部的请求&#xff0c;从而在UE中处理一些内容&#xff0c;但是之前只做过请求&#xff0c;哪整过这玩意&#xff0c;短期内还得出结果&#xff0c;那怎么搞嘞&#xff0c;本着省事的原则就找找呗&#xff0c;有没有现成的&#xff0…

Golang | Leetcode Golang题解之第123题买卖股票的最佳时机III

题目&#xff1a; 题解&#xff1a; func maxProfit(prices []int) int {buy1, sell1 : -prices[0], 0buy2, sell2 : -prices[0], 0for i : 1; i < len(prices); i {buy1 max(buy1, -prices[i])sell1 max(sell1, buy1prices[i])buy2 max(buy2, sell1-prices[i])sell2 m…

【Linux】进程间通信(System V IPC)

这节我们开始学习System V IPC方案。 分别是共享内存&#xff0c;消息队列与信号量 会着重讲解共享内存&#xff0c;但是消息队列与信号量只会说明一下原理。 原因&#xff1a;System V是新设计的一套标准 与文件的整合度不高只能进行本地通信 更何况&#xff0c;我们现在有…

IP代理池是什么?

从事跨境行业的朋友们总会有一个疑问&#xff0c;为什么自己所合作的IP代理商的IP在使用的过程中账号会有莫名封禁的问题&#xff0c;会不会是自己在使用的过程中错误的操作违反了平台的规则&#xff0c;其实不然有可能会是IP代理池纯净度不高的问题&#xff0c;有可能自己在使…

基于Jenkins+Kubernetes+GitLab+Harbor构建CICD平台

1. 实验环境 1.1 k8s环境 1&#xff09;Kubernetes 集群版本是 1.20.6 2&#xff09;k8s控制节点&#xff1a; IP&#xff1a;192.168.140.130 主机名&#xff1a;k8s-master 配置&#xff1a;4C6G 3&#xff09;k8s工作节点 节点1&#xff1a; IP&#xff1a;192.1…

基于字典树可视化 COCA20000 词汇

COCA20000 是美国当代语料库中最常见的 20000 个词汇&#xff0c;不过实际上有一些重复&#xff0c;去重之后大概是 17600 个&#xff0c;这些单词是很有用&#xff0c;如果能掌握这些单词&#xff0c;相信会对英语的能力有一个较大的提升。我很早就下载了这些单词&#xff0c;…

C++一个StringBad类

设计一个字符串类,下面的代码是一个不好的设计,起名StringBad。 //stringbad.h #pragma once //一个设计有问题的string类 #include <iostream> using namespace std;class StringBad { public:StringBad();//默认构造函数StringBad(const char* s);//构造函数~StringBa…

Java web应用性能分析之【jvisualvm远程连接云服务器】

Java web应用性能分析之【java进程问题分析概叙】-CSDN博客 Java web应用性能分析之【java进程问题分析工具】-CSDN博客 前面整理了java进程问题分析和分析工具&#xff0c;现在可以详细看看jvisualvm的使用&#xff0c;一般java进程都是部署云服务器&#xff0c;或者托管IDC机…

编译选项导致的结构体字节参数异常

文章目录 前言问题描述原因分析问题解决总结 前言 在构建编译工程时&#xff0c;会有一些对应的编译配置选项&#xff0c;不同的编译器&#xff0c;会有对应的配置项。本文介绍GHS工程中编译选项配置不对应导致的异常。 问题描述 在S32K3集成工程中&#xff0c;核1的INP_SWC…

【TB作品】MSP430F149,ADC采集,光强GY-30,DS18B20温度采集

功能 读取了GY-30 DS18B20 P6.0ADC P6.1ADC 显示到了LCD12864 硬件 //GY30 //SCL–P1.0 //SDA–P1.1 //VCC–3.3V //GND–GND //ADDR–不接 //DS18B20 //DATA–P1.6 //VCC–3.3V //GND–GND //ADC //DATA–P1.6 //P6.0 P6.1 ADC输入口 部分程序 #include <msp430.h>…

Java基础教程:算术运算符快速掌握

哈喽&#xff0c;各位小伙伴们&#xff0c;你们好呀&#xff0c;我是喵手。运营社区&#xff1a;C站/掘金/腾讯云&#xff1b;欢迎大家常来逛逛 今天我要给大家分享一些自己日常学习到的一些知识点&#xff0c;并以文字的形式跟大家一起交流&#xff0c;互相学习&#xff0c;一…

【Linux】文件系统和软硬链接

&#x1f466;个人主页&#xff1a;Weraphael ✍&#x1f3fb;作者简介&#xff1a;目前正在学习c和算法 ✈️专栏&#xff1a;Linux &#x1f40b; 希望大家多多支持&#xff0c;咱一起进步&#xff01;&#x1f601; 如果文章有啥瑕疵&#xff0c;希望大佬指点一二 如果文章对…

计算机毕业设计 | springboot+vue会议室管理系统(附源码)

1&#xff0c;绪论 1.1 项目背景 随着企业规模的不断扩大&#xff0c;会议室管理愈加复杂。传统的手工预约会议室的方式已经无法满足现代企业的需求&#xff0c;因此&#xff0c;开发一套会议室系统方案变得尤为重要。会议室系统可以实现会议室的在线预约、会议室资源的有效利…

每周统计-20240531

用于测试程序的稳定性&#xff1a; 龙虎榜&#xff1a; 成交额&#xff1a; 封成比&#xff1a; 收盘前放量&#xff1a; 开盘抢筹&#xff1a; 封单额&#xff1a;

Linux下的配置工具menuconfig+配置文件(Kconfig/.config/defconfig)

我们都知道,嵌入式开发中,或者说C语言中,配置基本都是通过宏定义来决定的,在MCU开发中,代码量比较小,配置项也比较少,我们直接修改对应的宏定义即可。 但是,Linux开发中,操作系统、驱动部分还有应用部分加起来,代码量极大,配置项目也非常多,这时候,就需要对这些配…

SSMP整合案例第五步 在前端页面上拿到service层调数据库里的数据后列表

在前端页面上列表 我们首先看看前端页面 我们已经把数据传入前端控制台 再看看我们的代码是怎么写的 我们展示 数据来自图dataList 在这里 我们要把数据填进去 就能展示在前端页面上 用的是前端数据双向绑定 axios发送异步请求 函数 //钩子函数&#xff0c;VUE对象初始化…

DiffBIR论文阅读笔记

这篇是董超老师通讯作者的一篇盲图像修复的论文&#xff0c;目前好像没看到发表在哪个会议期刊&#xff0c;应该是还在投&#xff0c;这个是arxiv版本&#xff0c;代码倒是开源了。本文所指的BIR并不是一个single模型对任何未知图像degradation都能处理&#xff0c;而是用同一个…