C++20 三路比较运算符

2024-04-08 05:38
文章标签 c++ 比较 运算符 20 三路

本文主要是介绍C++20 三路比较运算符,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

三路比较运算符

简单使用

基本数据类型已经已经内置定义了三路比较运算符,示例如下。

#include <iostream>using namespace std;int main()
{double var1 = 1.0;double var2 = 2.0;auto res = var1 <=> var2;if(res < 0)cout << "<" << endl;else if(res > 0)cout << ">" << endl;elsecout << "==" << endl;return 0;
}

三路比较运算表达式为值1 <=> 值2,可以简单的理解为当表达式为小于关系时,表达式返回负值,大于时返回正值,相等则为0。

三路比较运算的返回值。

具体内容可参考cppreference

如果操作数之一为 bool 类型而另一个不是,程序非良构。

若两个操作数均具有算术类型,或若一个具有无作用域枚举类型而另一个具有整型类型,则对各操作数应用一般算术转换,然后

若需要进行除了从整型类型到浮点类型之外的窄化转换,则程序非良构。 否则,若各操作数均具有整型类型,则运算符产出
std::strong_ordering 类型的纯右值: 若两个操作数算术上相等则为
std::strong_ordering::equal, 若第一操作数算术上小于第二个则为
std::strong_ordering::less 否则为 std::strong_ordering::greater。
否则,操作数均具有浮点类型,而运算符产出 std::partial_ordering 类型的纯右值。表达式 a <=> b 产出 若 a
小于 b 则为 std::partial_ordering::less, 若 a 大于 b 则为
std::partial_ordering::greater, 若 a 等价于 b 则为
std::partial_ordering::equivalent(-0 <=> +0 为等价), 否则为
std::partial_ordering::unordered(NaN <=> 任何值 为无序)。 若两个操作数都具有相同的枚举类型
E,则运算符产出将各操作数转换为 E 的底层类型再对转换后的操作数应用 <=> 的结果。

若至少一个操作数为指针或成员指针,则按需应用数组到指针转换、派生类到基类指针转换、函数指针转换和限定性转换,以将它们转换为同一指针类型,且结果指针类型为对象指针类型,则
p <=> q 返回 std::strong_ordering 类型的纯右值:

若 p == q 则为 std::strong_ordering::equal 若 q > p 则为
std::strong_ordering::less 若 p > q 则为 std::strong_ordering::greater。
若未指明这些指针值的比较(例如它们不指向同一对象或数组之中时),则为未指明的结果。 否则程序非良构。

以上内容来自"cppreference"

自定义类型使用三路比较运算符

一般的,可以使用编译器默认实现的三路比较运算符,示例如下。

#include <iostream>using namespace std;struct TestClass
{int var1;double var2;auto operator<=>(const TestClass&) const = default;
};int main()
{TestClass c1{ 2, 3.0 };TestClass c2{ 3, 2.0 };auto res = c1 <=> c2;if (res < 0)cout << "<" << endl;else if (res > 0)cout << ">" << endl;else if (res == 0)cout << "==" << endl;elsecout << "<>" << endl;return 0;
}

如果不使用默认实现,则可以手写其实现,示例如下。

#include <iostream>
#include <compare>using namespace std;struct TestClass
{int var1;double var2;auto operator<=>(const TestClass& other) const{if (other.var1 == var1 && other.var2 == var2)return std::partial_ordering::equivalent;if (other.var1 == var1 && other.var2 > var2 ||other.var1 > var1 && other.var2 == var2 ||other.var1 > var1 && other.var2 > var2)return std::partial_ordering::less;if (other.var1 == var1 && other.var2 < var2 ||other.var1 < var1 && other.var2 == var2 ||other.var1 < var1 && other.var2 < var2)return std::partial_ordering::greater;return std::partial_ordering::unordered;}
};int main()
{TestClass c1{ 2, 3.0 };TestClass c2{ 3, 2.0 };auto res = c1 <=> c2;if (res < 0)cout << "<" << endl;else if (res > 0)cout << ">" << endl;else if (res == 0)cout << "==" << endl;elsecout << "<>" << endl;return 0;
}

由上述两份代码的运行结果可以看出,自动生成的三路比较与我们自己手写的运行结果不一样,所以使用时需要注意,必须深刻理解默认生成的规则。

自动生成关系运算符

当定义了三路比较运算符后,也就相当于定义了其它比较运算符,例如operator ==是当三路比较运算返回std::partial_ordering::equivalent是为true其余情况为false,示例如下。

#include <iostream>using namespace std;struct TestClass
{int var1;double var2;auto operator<=>(const TestClass& other) const = default;
};int main()
{TestClass c1{ 2, 2.0 };TestClass c2{ 2, 2.0 };if (c1 == c2)cout << "==" << endl;if (c1 < c2)cout << "<" << endl;if (c1 > c2)cout << ">" << endl;return 0;
}

上述代码将会输出‘==’。

这篇关于C++20 三路比较运算符的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/884703

相关文章

利用c++判断水仙花数并输出示例代码

《利用c++判断水仙花数并输出示例代码》水仙花数是指一个三位数,其各位数字的立方和恰好等于该数本身,:本文主要介绍利用c++判断水仙花数并输出的相关资料,文中通过代码介绍的非常详细,需要的朋友可以... 以下是使用C++实现的相同逻辑代码:#include <IOStream>#include <vec

基于C++的UDP网络通信系统设计与实现详解

《基于C++的UDP网络通信系统设计与实现详解》在网络编程领域,UDP作为一种无连接的传输层协议,以其高效、低延迟的特性在实时性要求高的应用场景中占据重要地位,下面我们就来看看如何从零开始构建一个完整... 目录前言一、UDP服务器UdpServer.hpp1.1 基本框架设计1.2 初始化函数Init详解

C++ 右值引用(rvalue references)与移动语义(move semantics)深度解析

《C++右值引用(rvaluereferences)与移动语义(movesemantics)深度解析》文章主要介绍了C++右值引用和移动语义的设计动机、基本概念、实现方式以及在实际编程中的应用,... 目录一、右值引用(rvalue references)与移动语义(move semantics)设计动机1

C++ move 的作用详解及陷阱最佳实践

《C++move的作用详解及陷阱最佳实践》文章详细介绍了C++中的`std::move`函数的作用,包括为什么需要它、它的本质、典型使用场景、以及一些常见陷阱和最佳实践,感兴趣的朋友跟随小编一起看... 目录C++ move 的作用详解一、一句话总结二、为什么需要 move?C++98/03 的痛点⚡C++

Python中4大日志记录库比较的终极PK

《Python中4大日志记录库比较的终极PK》日志记录框架是一种工具,可帮助您标准化应用程序中的日志记录过程,:本文主要介绍Python中4大日志记录库比较的相关资料,文中通过代码介绍的非常详细,... 目录一、logging库1、优点2、缺点二、LogAid库三、Loguru库四、Structlogphp

详解C++ 存储二进制数据容器的几种方法

《详解C++存储二进制数据容器的几种方法》本文主要介绍了详解C++存储二进制数据容器,包括std::vector、std::array、std::string、std::bitset和std::ve... 目录1.std::vector<uint8_t>(最常用)特点:适用场景:示例:2.std::arra

C++构造函数中explicit详解

《C++构造函数中explicit详解》explicit关键字用于修饰单参数构造函数或可以看作单参数的构造函数,阻止编译器进行隐式类型转换或拷贝初始化,本文就来介绍explicit的使用,感兴趣的可以... 目录1. 什么是explicit2. 隐式转换的问题3.explicit的使用示例基本用法多参数构造

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C++打印 vector的几种方法小结

《C++打印vector的几种方法小结》本文介绍了C++中遍历vector的几种方法,包括使用迭代器、auto关键字、typedef、计数器以及C++11引入的范围基础循环,具有一定的参考价值,感兴... 目录1. 使用迭代器2. 使用 auto (C++11) / typedef / type alias

Python海象运算符:=的具体实现

《Python海象运算符:=的具体实现》海象运算符又称​​赋值表达式,Python3.8后可用,其核心设计是在表达式内部完成变量赋值并返回该值,从而简化代码逻辑,下面就来详细的介绍一下如何使用,感兴趣... 目录简介​​条件判断优化循环控制简化​推导式高效计算​正则匹配与数据提取​性能对比简介海象运算符