C++ 20新特性之三向比较运算符

2024-06-08 10:44

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

概述

        在C++中,如果需要对两个自定义类的对象进行比较,我们通常要单独定义6个比较运算符:==、!=、<、<=、>、>=。这不仅繁琐,还很容易出错,特别是当比较逻辑复杂时,稍有不慎就会引发不一致的比较结果。为了解决这个问题,提供统一的比较接口,C++ 20中引入了三向比较运算符。它能够根据操作数的相对大小,返回-1、0或1,分别代表小于、等于和大于,从而简化了比较逻辑的实现。

什么是三向比较运算符

        三向比较运算符,即<=>,通常称为“太空船运算符”(Spaceship operator)。这个运算符的设计初衷是为了简化用户自定义类型的比较操作,以前需要分别重载<、>、==等多个比较运算符,而现在仅需一个运算符就能完成所有比较逻辑的定义。

        <=>运算符在内部执行两个操作数的比较,并根据它们的相对大小返回一个特殊类型。这个类型属于std::compare_three_way的结果类型分类,具体可以是以下三种。

        std::strong_ordering:表示强顺序关系,返回std::strong_ordering::less、std::strong_ordering::equal、或std::strong_ordering::greater。

        std::weak_ordering:用于可能无法区分所有不同值的情况,比如NaN在浮点数比较中的处理。

        std::partial_ordering:适用于部分可比类型,比如某些情况下可能会出现不可比较的值。

如何使用

        对于简单的自定义类型,可以直接在类中使用= default默认定义<=>运算符,可参考下面的示例代码。

#include <compare>
#include <iostream>
using namespace std;struct Fraction
{int numerator;int denominator;Fraction(int num, int denom) : numerator(num), denominator(denom) {}// 重载三向比较运算符auto operator<=>(const Fraction& other) const = default;
};int main()
{Fraction f1(1, 2);Fraction f2(2, 4);if (f1 <=> f2 == 0){cout << "f1 and f2 are equal." << endl;}else if (f1 <=> f2 < 0){cout << "f1 is less than f2." << endl;}else{cout << "f1 is greater than f2." << endl;}return 0;
}

        在上面的示例代码中,operator<=>使用= default请求编译器生成比较逻辑。编译器会自动比较Fraction实例的numerator和denominator字段,如果两者都相等,则认为两个Fraction实例相等。如果numerator不等,则根据numerator的比较结果决定。如果numerator相同但denominator不等,则根据denominator的比较结果决定。

        有时候,默认生成的比较逻辑并不合适,就比如上面的例子。对于复杂的比较逻辑,可以手动实现<=>运算符,然后根据具体情况返回相应的比较结果。

        在下面的示例代码中,我们定义了一个Person结构体,并为它提供了三向比较运算符的重载函数。这个函数首先比较两个Person对象的年龄,如果年龄不同,则直接返回年龄的比较结果。如果年龄相同,则比较姓名。通过这种方式,我们可以很容易地实现自定义类型的比较功能,并且只需要一个比较函数就可以满足各种比较需求。

#include <compare>
#include <iostream>
#include <string>
using namespace std;struct Person
{std::string name;int age;auto operator<=>(const Person& other) const{// 先按年龄比较if (age != other.age){return age <=> other.age;}// 年龄相同则按姓名比较return name <=> other.name;}
};int main()
{Person p1{ "Mike", 18 };Person p2{ "Jack", 20 };std::cout << std::boolalpha;std::cout << (p1 < p2) << endl;std::cout << (p1 > p2) << endl;// 使用三向比较运算符的返回值  auto result = p1 <=> p2;if (result < 0){std::cout << "p1 is less than p2" << endl;}else if (result == 0){std::cout << "p1 is equal to p2" << endl;}else{std::cout << "p1 is greater than p2" << endl;}return 0;
}

总结

        三向比较运算符为我们提供了更加直观、简洁的比较方式,使得代码更加优雅、更易于维护。通过提供一个统一的比较接口,它减少了代码量,提高了代码的可读性和可维护性。对于需要实现自定义类型比较功能的软件开发者来说,这个新特性无疑是一个巨大的福音。

💡 如果想阅读最新的文章,或者有技术问题需要交流和沟通,可搜索并关注微信公众号“希望睿智”。

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



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

相关文章

C++中实现调试日志输出

《C++中实现调试日志输出》在C++编程中,调试日志对于定位问题和优化代码至关重要,本文将介绍几种常用的调试日志输出方法,并教你如何在日志中添加时间戳,希望对大家有所帮助... 目录1. 使用 #ifdef _DEBUG 宏2. 加入时间戳:精确到毫秒3.Windows 和 MFC 中的调试日志方法MFC

五大特性引领创新! 深度操作系统 deepin 25 Preview预览版发布

《五大特性引领创新!深度操作系统deepin25Preview预览版发布》今日,深度操作系统正式推出deepin25Preview版本,该版本集成了五大核心特性:磐石系统、全新DDE、Tr... 深度操作系统今日发布了 deepin 25 Preview,新版本囊括五大特性:磐石系统、全新 DDE、Tree

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

百度/小米/滴滴/京东,中台架构比较

小米中台建设实践 01 小米的三大中台建设:业务+数据+技术 业务中台--从业务说起 在中台建设中,需要规范化的服务接口、一致整合化的数据、容器化的技术组件以及弹性的基础设施。并结合业务情况,判定是否真的需要中台。 小米参考了业界优秀的案例包括移动中台、数据中台、业务中台、技术中台等,再结合其业务发展历程及业务现状,整理了中台架构的核心方法论,一是企业如何共享服务,二是如何为业务提供便利。

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象