news 2026/4/2 23:26:12

C++编程实践——std::void_t的实践应用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C++编程实践——std::void_t的实践应用

一、介绍

在前面已经分析过C++17中的std::void_t,对其已经有了一个整体的认知,同时对标准中对std::void_t的相关技术细节进行了分析说明。虽然对其的应用也进行了初步的分析,但如何在实际的工程实践中进行应用以及其应用的场景,还需要进一步的展开。
先回忆一下标准库的实现:

template<class...>usingvoid_t=void;

它采用变参模板,将任意数量的类型参数映射为 void 类型。
不过对于刚刚接触模板技术的开发者来说,可能不好理解,可以按下面的方式来看就容易理解了:

template<typename...Ts>structmake_void{using type=void;};template<typename...Ts>usingvoid_t=typename make_void<Ts...>::type;

当模板被实例化时,make_void<Ts…>会让其内部成员type始终是void类型。其本质也是依赖于SFINAE技术。即模板参数列表中的类型或表达式(比如decltype(…))无效时,std::void_t的别名定义会无法定义成功,进而触发 SFINAE,将该特化模板从候选模板中移除,不会引发编译错误。

二、工程实践的应用方式

在实际的工程实践中,应用的场景很多,但应用的方法主要包括:

  1. 做为默认值
    模板开发中,经常会遇到默认参数的情况:
template<typename T,typename U=std::void_t<>>structDemo;
  1. SFINAE的触发
    与decltype等配合使用进行SFINAE的触发,比如下面的情况:
// 通用模板template<typename T,typename=void>class Demo{...};template<typename T>class Demo<T,std::void_t<decltype(std::declval<T>().try()),decltype(std::declval<T>().tryTest())>>{...};
  1. 模板的特化处理
    利用std::void_t进行模板的偏特化:
// size() 成员函数检测template<typename T,typename=void>structcheckSize:std::false_type{};template<typename T>structcheckSize<T,std::void_t<decltype(std::declval<T>().size())>>:std::true_type{};

std::void_t应用非常广泛,它可以在SFINAE的应用中大大减化相关的复杂度,同时,在某些C++20概念无法应用的场景下它又可以起到某些替换的能力 。虽然上面的总结有些粗浅,但也是从另外一个角度对std::void_t的应用进行一个初步的尝试。大家可以继续在此基础上进行补充,加强自己的学习和总结能力。

三、应用场景

虽然在前面的std::void_t说明分析中对其应用进行了初步的分析总结,但条理性不太清楚,这次再整体总结一下:

  1. 类型特性的单一检测
    如常见的属性、函数以及相关特定的操作(如操作运算符、智能指针和嵌套等)等
  2. 多类型特性的检测
    即对属性、函数等进行多项控制检测
  3. SFINAE的应用
    即将std::void_t应用于复杂的SFINAE技术应用中,如表达式的有效性等
  4. 实现Conecpts
    这种一般属于开发者在早期版本中想应用一些类似概念的实现

四、例程

看一下相关应用例程:

#include<iostream>#include<type_traits>// 检测类型别名template<typename,typename=std::void_t<>>structcheck_type_member:std::false_type{};template<typename T>structcheck_type_member<T,std::void_t<typename T::type>>:std::true_type{};// 检测成员变量template<typename,typename=std::void_t<>>structcheck_member:std::false_type{};template<typename T>structcheck_member<T,std::void_t<decltype(T::data)>>:std::true_type{};// 检测成员函数template<typename,typename=std::void_t<>>structcheck_display:std::false_type{};template<typename T>structcheck_display<T,std::void_t<decltype(std::declval<T>().display())>>:std::true_type{};// 检测智能指针template<typename,typename=std::void_t<>>structis_smart_pointer:std::false_type{};template<typename T>structis_smart_pointer<T,std::void_t<decltype(std::declval<T>().operator->()),decltype(std::declval<T>().get())>>:std::true_type{};structDemoType{using type=int;};structDemoNoType{};structDemoMember{intdata;};structDemoNoMember{};structDemoDisplay{voiddisplay(){}};structDemoNoDisplay{};structDemoSmartPtr{int*operator->();void*get();};structDemoNotSmart{};intmain(){std::cout<<"type is exist: "<<check_type_member<DemoType>::value<<std::endl;std::cout<<"type is exist: "<<check_type_member<DemoNoType>::value<<std::endl;std::cout<<"Does member data exist: "<<check_member<DemoMember>::value<<std::endl;std::cout<<"Does member data exist: "<<check_member<DemoNoMember>::value<<std::endl;std::cout<<"function:display is: "<<check_display<DemoDisplay>::value<<std::endl;std::cout<<"function:display is: "<<check_display<DemoNoDisplay>::value<<std::endl;std::cout<<"SmartPtr :is a smart pointer: "<<is_smart_pointer<DemoSmartPtr>::value<<std::endl;std::cout<<"NotSmart: is a smart pointer:"<<is_smart_pointer<DemoNotSmart>::value<<std::endl;return0;}

前面的is_detected的实现也利用了std::void_t,大家可以回过头去看看,这样可以互相印证,加深理解。另外如果想进行联合检测除了使用传统的方法(如SFINAE)外也可以使用std::conjunction来进行多个状态值的整体判断。
在看明白上述的代码后,就可以从网上找一段复杂的相关的例子自行拆解,会发现很容易就明白了开发者编写的目的。

五、总结

std::void_t虽然是在C++17中引入的,但其实在早期的版本中就有类似的实践。大家可以理解为std::void_t是对SFINAE技术和模板特化的一种标准化处理。通过对std::void_t的应用,大家可以从中学习一些相关的开发技巧,引入到自己的开发中去。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/3/31 5:00:55

PingFang SC Regular字体资源下载指南

PingFang SC Regular字体资源下载指南 【免费下载链接】PingFangSCRegular字体资源下载 探索PingFang SC Regular字体的魅力&#xff0c;这是一套专为现代设计和开发需求打造的中文字体。本资源库提供了多种格式的字体文件&#xff0c;包括eot、otf、svg、ttf和woff&#xff0c…

作者头像 李华
网站建设 2026/3/26 14:38:19

如何快速掌握C4编译器:面向新手的终极指南

如何快速掌握C4编译器&#xff1a;面向新手的终极指南 【免费下载链接】c4 x86 JIT compiler in 86 lines 项目地址: https://gitcode.com/gh_mirrors/c42/c4 C4编译器是一个极简主义的杰作&#xff0c;它用仅仅四个函数就实现了完整的C语言编译功能。这个开源项目不仅展…

作者头像 李华
网站建设 2026/3/21 7:09:07

高效微信管理神器:WeChatTweak-macOS防撤回与多开功能完全指南

高效微信管理神器&#xff1a;WeChatTweak-macOS防撤回与多开功能完全指南 【免费下载链接】WeChatTweak-macOS A dynamic library tweak for WeChat macOS - 首款微信 macOS 客户端撤回拦截与多开 &#x1f528; 项目地址: https://gitcode.com/gh_mirrors/we/WeChatTweak-m…

作者头像 李华
网站建设 2026/3/31 22:00:45

Ink/Stitch免费刺绣设计软件完整使用指南

Ink/Stitch免费刺绣设计软件完整使用指南 【免费下载链接】inkstitch Ink/Stitch: an Inkscape extension for machine embroidery design 项目地址: https://gitcode.com/gh_mirrors/in/inkstitch 厌倦了昂贵的专业刺绣软件&#xff1f;想要一个真正免费且功能强大的设…

作者头像 李华
网站建设 2026/3/28 1:05:10

Simditor多语言解决方案:构建全球化富文本编辑器的技术实践

Simditor多语言解决方案&#xff1a;构建全球化富文本编辑器的技术实践 【免费下载链接】simditor An Easy and Fast WYSIWYG Editor 项目地址: https://gitcode.com/gh_mirrors/si/simditor 在数字内容创作日益全球化的今天&#xff0c;富文本编辑器作为内容生产的核心…

作者头像 李华
网站建设 2026/3/29 2:08:43

Python版本管理终极指南:告别版本冲突,拥抱高效开发

Python版本管理终极指南&#xff1a;告别版本冲突&#xff0c;拥抱高效开发 【免费下载链接】pyenv Simple Python version management 项目地址: https://gitcode.com/GitHub_Trending/py/pyenv 你是否曾经遇到过这样的情况&#xff1a;新项目需要Python 3.11的最新特性…

作者头像 李华