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++ RabbitMq消息队列组件详解

《C++RabbitMq消息队列组件详解》:本文主要介绍C++RabbitMq消息队列组件的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. RabbitMq介绍2. 安装RabbitMQ3. 安装 RabbitMQ 的 C++客户端库4. A

C++ HTTP框架推荐(特点及优势)

《C++HTTP框架推荐(特点及优势)》:本文主要介绍C++HTTP框架推荐的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. Crow2. Drogon3. Pistache4. cpp-httplib5. Beast (Boos

JDK9到JDK21中值得掌握的29个实用特性分享

《JDK9到JDK21中值得掌握的29个实用特性分享》Java的演进节奏从JDK9开始显著加快,每半年一个新版本的发布节奏为Java带来了大量的新特性,本文整理了29个JDK9到JDK21中值得掌握的... 目录JDK 9 模块化与API增强1. 集合工厂方法:一行代码创建不可变集合2. 私有接口方法:接口

C#特性(Attributes)和反射(Reflection)详解

《C#特性(Attributes)和反射(Reflection)详解》:本文主要介绍C#特性(Attributes)和反射(Reflection),具有很好的参考价值,希望对大家有所帮助,如有错误... 目录特性特性的定义概念目的反射定义概念目的反射的主要功能包括使用反射的基本步骤特性和反射的关系总结特性

PyTorch高级特性与性能优化方式

《PyTorch高级特性与性能优化方式》:本文主要介绍PyTorch高级特性与性能优化方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、自动化机制1.自动微分机制2.动态计算图二、性能优化1.内存管理2.GPU加速3.多GPU训练三、分布式训练1.分布式数据

C++类和对象之初始化列表的使用方式

《C++类和对象之初始化列表的使用方式》:本文主要介绍C++类和对象之初始化列表的使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C++初始化列表详解:性能优化与正确实践什么是初始化列表?初始化列表的三大核心作用1. 性能优化:避免不必要的赋值操作2. 强

C++迭代器失效的避坑指南

《C++迭代器失效的避坑指南》在C++中,迭代器(iterator)是一种类似指针的对象,用于遍历STL容器(如vector、list、map等),迭代器失效是指在对容器进行某些操作后... 目录1. 什么是迭代器失效?2. 哪些操作会导致迭代器失效?2.1 vector 的插入操作(push_back,

Kotlin运算符重载函数及作用场景

《Kotlin运算符重载函数及作用场景》在Kotlin里,运算符重载函数允许为自定义类型重新定义现有的运算符(如+-…)行为,从而让自定义类型能像内置类型那样使用运算符,本文给大家介绍Kotlin运算... 目录基本语法作用场景类对象数据类型接口注意事项在 Kotlin 里,运算符重载函数允许为自定义类型重

C#如何调用C++库

《C#如何调用C++库》:本文主要介绍C#如何调用C++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

SpringBoot3.4配置校验新特性的用法详解

《SpringBoot3.4配置校验新特性的用法详解》SpringBoot3.4对配置校验支持进行了全面升级,这篇文章为大家详细介绍了一下它们的具体使用,文中的示例代码讲解详细,感兴趣的小伙伴可以参考... 目录基本用法示例定义配置类配置 application.yml注入使用嵌套对象与集合元素深度校验开发