news 2026/4/13 8:15:29

跟我学C++中级篇—C++17中的元编程逻辑操作

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
跟我学C++中级篇—C++17中的元编程逻辑操作

一、逻辑操作

在C++中,逻辑运算符算是最常见的一种运算符,如&&,||以及!。这三种逻辑运算符对于处理条件判断和循环控制等有着重要的作用。说的更简单一些,就是处理程序的分支路径。这也符合现实世界中的工作处理实践。毕竟,计算机程序本来的目的就是为了解决现实世界中的问题。

二、元编程的逻辑操作

随着泛型编程特别是其中的元编程的发展,在编译期进行计算的需求也成为了一个重要的方向。&&这类逻辑运算符一般是只能用于运行时的逻辑操作,虽然在某些情况下(如编译时的常量表达式的逻辑运算)也可以使用。但在更多的情况下还是在运行时进行处理逻辑操作。
理论上讲,把它们扩展到编译期也不是不可能的。但是,从设计原则上和发展的眼光来看,这是一种非常不合理的设计。所以在C++17中,针对元编程中的逻辑操作给出了相应的三个逻辑操作的模板即:std::conjunction、std::disjunction和std::negation。即逻辑与,逻辑或和逻辑非。

三、std::conjunction、std::disjunction和std::negation

下面将针对这三个逻辑模板进行分析:

  1. std::conjunction
    逻辑与的操作,它接受N个布尔类型如前面提到的std::true_type或 std::false_type,它同样具有短路的行为,一量遇到std::false_type,将会终止模板类型的实例化。其也提供了 std::conjunction_v这种辅助变量模板。
#include<iostream>#include<type_traits>// func is enabled if all Ts... have the same type as Ttemplate<typename T,typename...Ts>std::enable_if_t<std::conjunction_v<std::is_same<T,Ts>...>>func(T,Ts...){std::cout<<"All types in pack are the same.\n";}// otherwisetemplate<typename T,typename...Ts>std::enable_if_t<!std::conjunction_v<std::is_same<T,Ts>...>>func(T,Ts...){std::cout<<"Not all types in pack are the same.\n";}template<typename T,typename...Ts>constexpr bool all_types_are_same=std::conjunction_v<std::is_same<T,Ts>...>;static_assert(all_types_are_same<int,int,int>);static_assert(not all_types_are_same<int,int&,int>);intmain(){func(1,2,3);func(1,2,"hello!");}
  1. std::disjunction
    逻辑或操作,它提供对多个布尔类型的执行或操作,同样它也支持短路行为,一但有一个std::true_type,后续将不再进行模板的实例化。其同样提供了disjunction_v这个辅助模板。
#include<cstdint>#include<string>#include<type_traits>// values_equal<a, b, T>::value is true if and only if a == b.template<autoV1,decltype(V1)V2,typename T>structvalues_equal:std::bool_constant<V1==V2>{using type=T;};// default_type<T>::value is always truetemplate<typename T>structdefault_type:std::true_type{using type=T;};// Now we can use disjunction like a switch statement:template<intI>using int_of_size=typename std::disjunction<//values_equal<I,1,std::int8_t>,//values_equal<I,2,std::int16_t>,//values_equal<I,4,std::int32_t>,//values_equal<I,8,std::int64_t>,//default_type<void>// must be last!>::type;static_assert(sizeof(int_of_size<1>)==1);static_assert(sizeof(int_of_size<2>)==2);static_assert(sizeof(int_of_size<4>)==4);static_assert(sizeof(int_of_size<8>)==8);static_assert(std::is_same_v<int_of_size<13>,void>);// checking if Foo is constructible from double will cause a hard errorstructFoo{template<class T>structsfinae_unfriendly_check{static_assert(!std::is_same_v<T,double>);};template<class T>Foo(T,sfinae_unfriendly_check<T>={});};template<class...Ts>structfirst_constructible{template<class T,class...Args>structis_constructible_x:std::is_constructible<T,Args...>{using type=T;};structfallback{staticconstexpr bool value=true;using type=void;// type to return if nothing is found};template<class...Args>using with=typename std::disjunction<is_constructible_x<Ts,Args...>...,fallback>::type;};// OK, is_constructible<Foo, double> not instantiatedstatic_assert(std::is_same_v<first_constructible<std::string,int,Foo>::with<double>,int>);static_assert(std::is_same_v<first_constructible<std::string,int>::with<>,std::string>);static_assert(std::is_same_v<first_constructible<std::string,int>::with<constchar*>,std::string>);static_assert(std::is_same_v<first_constructible<std::string,int>::with<void*>,void>);intmain(){}
  1. std::negation
    逻辑非操作,主要用于对模板参数B进行逻辑否定。同样也有negation_v辅助模板。
#include<type_traits>static_assert(std::is_same<std::bool_constant<false>,typename std::negation<std::bool_constant<true>>::type>::value,"");static_assert(std::is_same<std::bool_constant<true>,typename std::negation<std::bool_constant<false>>::type>::value,"");static_assert(std::negation_v<std::bool_constant<true>>==false);static_assert(std::negation_v<std::bool_constant<false>>==true);intmain(){}

*上述代码均来自cppreference的示例代码
它们三个主要可以用于以下场景:

  1. 条件控制和处理
    可以与 std::enable_if_t或std::find_if等进行条件编译、判断和分支控制,实现SFINAE的处理。如:
template<typename T>std::enable_if_t<std::conjunction_v<std::is_integral<T>,std::is_signed<T>>,void>Check(T value){// 控制重载}
  1. 复杂类型或自定义类型
    可以通过其实现复杂类型的判断或自定义实现相关的数据类型,如:
template<typename T>using is_owner_integral=std::conjunction<std::is_integral<T>,std::is_signed<T>>;

包括模板的特化处理等,都可以通过其来控制
3. 编译错误控制和接口约束
通过逻辑判断(包括static_assert),如短路等来避免对更深层的复杂模板编译处理

四、两类逻辑操作的区别

在上面的分析中,发现二者虽然都可以实现逻辑的操作,但在具体应用中还是有所不同,主要包括:

  1. 应用时期不同
    元编程中的逻辑操作用于编译期;而逻辑运算符则在运行期应用(虽然在特定情况下逻辑操作运算符也可以用在编译期)
  2. 面向的操作对象不同
    元编程的逻辑操针对类型(类型特征模板);而逻辑运算符针对的布尔值
  3. 运算结果不同
    元编程的逻辑操作返回的是std::true_type等类型;而逻辑运算符返回的是布尔值
  4. 面对场景不同
    元编程的逻辑操作主要用于模板元编程、SFINAE等编译期的条件控制;而逻辑运算符则用于逻辑判断
  5. 性能和复杂度
    元编程的逻辑操作可能增加了编译时间,应用较为复杂,但提供了清晰可读性语法;而逻辑运算符直接编译成机器指令在运行时操作,开销低

五、总结

C++语言也是不断的发展和完善的,随着元编程技术被标准不断推进,相关的配套模块也会不断的充实和完善。这就需要开发者不断的跟进标准的发展,不断的学习并及时的整合到自己的知识体系中。

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

57350001-CU DSAV111视频导体模块

57350001-CU DSAV111 视频导体模块&#xff1a;专用于工业自动化系统的视频信号传输支持多路高清模拟或数字视频信号内置抗干扰设计&#xff0c;保证信号稳定清晰模块化结构&#xff0c;便于快速安装与更换提供过压、短路及过温保护支持远距离信号传输&#xff0c;保持图像质量…

作者头像 李华
网站建设 2026/3/31 0:09:57

57310001-KD DSBC172总线重复模块

57310001-KD DSBC172 总线重复模块&#xff1a;用于工业控制系统总线信号的中继与扩展提升总线通信距离&#xff0c;确保数据稳定传输支持多种现场总线协议&#xff0c;兼容性强内置信号整形与隔离功能&#xff0c;抗干扰能力高模块化设计&#xff0c;便于快速安装和维护提供状…

作者头像 李华
网站建设 2026/4/9 15:35:40

Java爬虫的几种方式

Java多线程与高并发 一、基本概念 1.0 字节和位的关系 一个字节&#xff08;Byte&#xff09;是计算机存储和传输数据的基本单位&#xff0c;它表示8个二进制位&#xff08;bits&#xff09;。因此&#xff0c;1个字节等于8个比特&#xff08;bits&#xff09;。 当谈到计算…

作者头像 李华
网站建设 2026/4/2 9:55:56

Qwen2.5-7B性能全解析|长文本生成与多语言支持实测

Qwen2.5-7B性能全解析&#xff5c;长文本生成与多语言支持实测 引言&#xff1a;为何关注Qwen2.5-7B&#xff1f; 在大模型快速迭代的今天&#xff0c;长上下文理解能力和多语言泛化表现已成为衡量一个语言模型是否具备工业级应用潜力的关键指标。阿里云最新发布的 Qwen2.5-7…

作者头像 李华
网站建设 2026/4/1 22:46:12

ResNet18蚂蚁蜜蜂分类:新手友好教程,没GPU也能学AI

ResNet18蚂蚁蜜蜂分类&#xff1a;新手友好教程&#xff0c;没GPU也能学AI 引言&#xff1a;生物研究的AI助手 作为一名生物专业的研究者&#xff0c;你是否经常需要处理大量昆虫图像数据&#xff1f;传统的人工分类方法不仅耗时耗力&#xff0c;还容易因疲劳导致误判。今天我…

作者头像 李华