news 2026/6/15 18:33:01

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/6/15 15:16:00

教你使用Dify搭建爬虫工作流

最近用Dify做了一个工作流应用&#xff0c;可以实现自动化采集亚马逊商品信息&#xff0c;包括名称、价格、折扣、评分、评论等关键字段&#xff0c;然后使用DeepSeek对商品竞争力、价格、用户口碑进行分析&#xff0c;为跨境卖家提供一份完整的分析报告。 整个工作流搭建用到了…

作者头像 李华
网站建设 2026/6/15 11:25:57

279. Java Stream API - Stream 拼接的两种方式:concat() vs flatMap()

文章目录 279. Java Stream API - Stream 拼接的两种方式&#xff1a;concat() vs flatMap()&#x1f3af; 本节目标 &#x1f9ea; 背景问题&#xff1a;多个集合如何组合成一个流&#xff1f;✅ 方式一&#xff1a;使用 Stream.concat()&#x1f4cc; 特点&#xff1a;&#…

作者头像 李华
网站建设 2026/6/15 12:34:45

终极私有云存储网盘:JmalCloud让您的数据安全又高效

终极私有云存储网盘&#xff1a;JmalCloud让您的数据安全又高效 【免费下载链接】jmal-cloud-view JmalCloud 是一款私有云存储网盘项目&#xff0c;能够简单安全管理您的云端文件 项目地址: https://gitcode.com/gh_mirrors/jm/jmal-cloud-view 在数据安全日益重要的今…

作者头像 李华
网站建设 2026/6/15 12:53:40

AR.js开发终极指南:快速搭建Web增强现实应用

AR.js开发终极指南&#xff1a;快速搭建Web增强现实应用 【免费下载链接】AR.js Efficient Augmented Reality for the Web - 60fps on mobile! 项目地址: https://gitcode.com/gh_mirrors/ar/AR.js 想要在网页中实现令人惊叹的增强现实效果吗&#xff1f;AR.js让你无需…

作者头像 李华
网站建设 2026/6/15 12:41:05

使用DeepSpeed ZeRO3进行超大规模模型训练的最佳实践

使用DeepSpeed ZeRO3进行超大规模模型训练的最佳实践 在当今大模型时代&#xff0c;训练一个千亿参数的模型早已不再是“有没有算力”的问题&#xff0c;而是“如何用有限资源高效完成训练”的工程挑战。当你面对一台装有8张A100的服务器&#xff0c;却想微调Qwen-70B或LLaMA3-…

作者头像 李华