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.mzph.cn/web/20950.shtml

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

相关文章

猫头虎分享已解决Bug || **Eslint插件安装问题Unable to resolve eslint-plugin-猫头虎

猫头虎分享已解决Bug || **Eslint插件安装问题Unable to resolve eslint-plugin-猫头虎 博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a; 《面试题大全》 — 面试准备的…

将 py 文件编译成 pyd 文件

文章目录 一、简介1.1、Python中的文件类型&#xff1a;.py .pyc .pyd1.2、基本原理1.2.1、函数详解&#xff1a;Extension() —— 用于定义扩展模块&#xff08;C/C 扩展&#xff09;的类1.2.2、函数详解&#xff1a;setup() —— 用于配置和构建包的函数 二、构建过程2.0、…

百度文心一言API批量多线程写文章软件-key免费无限写

百度文心大模型的两款主力模型ENIRE Speed、ENIRE Lite全面免费&#xff0c;即刻生效。 百度文心大模型的两款主力模型 这意味着&#xff0c;大模型已进入免费时代&#xff01; 据了解&#xff0c;这两款大模型发布于今年 3 月&#xff0c;支持 8K 和 128k 上下文长度。 ER…

Java集合面试题(概述,list,Map)

一个常见的"fail-safe"集合例子是CopyOnWriteArrayList。这个集合在每次修改时都会复制当前的数组&#xff0c;修改操作在新数组上进行&#xff0c;而遍历操作则在旧数组上进行。这样&#xff0c;即使在遍历过程中进行了修改&#xff0c;也不会影响遍历的进行。 插入…

车载诊断内容汇总(培训+视频)

车载诊断内容汇总 我是穿拖鞋的汉子&#xff0c;魔都中坚持长期主义的汽车电子工程师。 老规矩&#xff0c;分享一段喜欢的文字&#xff0c;避免自己成为高知识低文化的工程师&#xff1a; 屏蔽力是信息过载时代一个人的特殊竞争力&#xff0c;任何消耗你的人和事&#xff0c…

pyopengl 立方体 正投影,透视投影

目录 顶点和线的方式 划线的方式实现: 顶点和线的方式 import numpy as np from PyQt5 import QtWidgets from PyQt5.QtCore import Qt from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton from OpenGL.GL import * from OpenGL.GLU import * import sys…

Java大文件上传、分片上传、多文件上传、断点续传、上传文件minio、分片上传minio等解决方案

一、上传说明 文件上传花样百出&#xff0c;根据不同场景使用不同方案进行实现尤为必要。通常开发过程中&#xff0c;文件较小&#xff0c;直接将文件转化为字节流上传到服务器&#xff0c;但是文件较大时&#xff0c;用普通的方法上传&#xff0c;显然效果不是很好&#xff0c…

【Unity脚本】修改游戏对象的活动状态

【知识链】Unity -> Unity脚本 -> 游戏对象 -> 活动状态【摘要】本文介绍了如何通过编辑器和脚本来访问游戏对象的活动状态&#xff0c;并给出具体的场景示例。 文章目录 第一章 引言第二章 在编辑器中设置活动状态2.1. 在编辑器中设置活动状态2.1.1. 停用游戏对象2.…

文件IO(三)

文件IO&#xff08;三&#xff09; 左移右移Linux的man 手册文件IO打开文件操作文件关闭文件 caps lock开灯关灯读取按键文件IO操作目录文件打开目录文件操作目录文件 库动态库和静态库的优缺点创建静态库创建动态库 按下右ctrl键 亮灭灯 左移右移 Linux的man 手册 文件IO 打开…

FJSP:常春藤算法(Ivy algorithm,LVYA)求解柔性作业车间调度问题(FJSP),提供MATLAB代码

详细介绍 FJSP&#xff1a;常春藤算法&#xff08;Ivy algorithm&#xff0c;LVYA&#xff09;求解柔性作业车间调度问题&#xff08;FJSP&#xff09;&#xff0c;提供MATLAB代码-CSDN博客 完整MATLAB代码 FJSP&#xff1a;常春藤算法&#xff08;Ivy algorithm&#xff0c;…

图形学初识--多边形剪裁算法

文章目录 前言正文为什么需要多边形剪裁算法&#xff1f;前置知识二维直线直线方程&#xff1a;距离本质&#xff1a;点和直线距离关系&#xff1a; 三维平面平面方程距离本质&#xff1a;点和直线距离关系&#xff1a; Suntherland hodgman算法基本介绍基本思想二维举例问题描…

最小时间差

首先可以想到&#xff0c;可以计算出任意两个时间之间的差值&#xff0c;然后比较出最小的&#xff0c;不过这种蛮力方法时间复杂度是O(n^2)。而先将时间列表排序&#xff0c;再计算相邻两个时间的差值&#xff0c;就只需要计算n个差值&#xff0c;而排序阶段时间复杂度通常为O…

C语言实现贪吃蛇小游戏(控制台)

本篇主要内容是使用C语言在Windows环境的控制台中模拟实现经典小游戏贪吃蛇。 一、准备工作 我们要实现的基本功能有&#xff1a; 地图绘制蛇吃食物的功能&#xff08;上、下、左、右方向键控制蛇的动作&#xff09;蛇撞墙死亡蛇撞自身死亡计算得分蛇身加速、减速暂停游戏 …

9-Django项目--验证码操作

目录 templates/login/login.html utils/code.py views/login.py 验证码 生成验证码 code.py 应用验证码 views.py login.html templates/login/login.html {% load static %} <!DOCTYPE html> <html lang"en"> <head><meta charset&q…

PID算法入门

文章目录 122.12.22.3 344.14.24.3 1 e(t) 是偏差 实 和 目u(t) 是运算结果 2 层层叠加 得出完整的离散公式 2.1 kp 越大 系统偏差 减小的越快kp大的时候 会出现过冲现象&#xff1f; 0.5 那个会快他解释过冲 &#xff1a; 0.2的 5分钟正好到了 那0.5的五分钟 升的就比20多 就…

④单细胞学习-cellchat细胞间通讯

目录 1&#xff0c;原理基础 流程 受体配体概念 方法比较 计算原理 2&#xff0c;数据 3&#xff0c;代码运行 1&#xff0c;原理基础 原文学习Inference and analysis of cell-cell communication using CellChat - PMC (nih.gov) GitHub - sqjin/CellChat: R toolk…

在 JavaScript 中实现数据加密与解密:Web Cryptography API 与 CryptoJS详解

在 JavaScript 中&#xff0c;可以使用 Web Cryptography API 或第三方库如 crypto-js 来实现加密和解密。本文将介绍如何使用这两种方法在客户端进行数据的加密和解密。 使用 Web Cryptography API Web Cryptography API 是现代浏览器提供的一个强大、原生的加密 API。它允许…

关于留痕的使用常见的问题

1. 登录微信 登录要导出数据的微信&#xff08;不支持微信多开&#xff0c;不支持部分老版本微信&#xff09; 相关信息 想把手机端的微信聊天记录转移到电脑上可以使用微信自带的聊天记录迁移功能 操作步骤&#xff1a; 安卓&#xff1a; 手机微信->我->设置->聊…

[深度学习]使用python部署yolov10的onnx模型

测试环境&#xff1a; onnxruntime1.15.1 opencv-python4.8.0.76 部分实现代码&#xff1a; parser argparse.ArgumentParser()parser.add_argument("--model", typestr, default"yolov10n.onnx", help"Input your ONNX model.")parser.add_arg…

电子电器架构 --- 什么是域控制器?

我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明自己,无利益不试图说服别人,是精神上的节…