0

0

C++如何实现对象之间的比较操作

P粉602998670

P粉602998670

发布时间:2025-09-16 08:45:01

|

182人浏览过

|

来源于php中文网

原创

通过运算符重载实现C++对象比较,核心是定义operator==和operator

c++如何实现对象之间的比较操作

在C++中,实现对象之间的比较操作,核心思路就是通过运算符重载来定义对象之间“相等”、“小于”等关系的逻辑。这通常涉及重载

operator==
(相等)和
operator<
(小于),因为有了这两个基础,其他比较运算符(如
!=
>
<=
>=
)往往可以根据它们推导出来,或者在C++20及以后版本中,通过三路比较运算符
operator<=>
(飞船运算符)一劳永逸地解决。

解决方案

要让C++自定义类型的对象能够像基本类型那样进行比较,我们必须明确告诉编译器“比较”对于我们的对象意味着什么。最直接且常用的方式就是重载比较运算符。

1. 重载

operator==
operator<
(C++17及以前)

这是最基础也最灵活的方法。通常,我们会选择重载这两个运算符,因为它们是许多标准库算法和容器(如

std::sort
std::map
std::set
)所依赖的。

立即学习C++免费学习笔记(深入)”;

  • operator==
    (相等):定义两个对象何时被认为是相等的。

    #include 
    #include 
    
    class Person {
    public:
        std::string name;
        int age;
    
        Person(std::string n, int a) : name(std::move(n)), age(a) {}
    
        // 作为成员函数重载 operator==
        bool operator==(const Person& other) const {
            return name == other.name && age == other.age;
        }
    
        // 作为成员函数重载 operator<
        // 定义排序规则:先按年龄,年龄相同则按姓名
        bool operator<(const Person& other) const {
            if (age != other.age) {
                return age < other.age;
            }
            return name < other.name;
        }
    
        // 辅助输出,方便调试
        friend std::ostream& operator<<(std::ostream& os, const Person& p) {
            return os << "Person(" << p.name << ", " << p.age << ")";
        }
    };
    
    // 如果不想作为成员函数,也可以作为非成员函数重载
    // 此时可能需要访问私有成员,可以声明为friend
    /*
    bool operator==(const Person& lhs, const Person& rhs) {
        return lhs.name == rhs.name && lhs.age == rhs.age;
    }
    bool operator<(const Person& lhs, const Person& rhs) {
        if (lhs.age != rhs.age) {
            return lhs.age < rhs.age;
        }
        return lhs.name < rhs.name;
    }
    */
    
    // 其他比较运算符可以基于 == 和 < 来实现
    bool operator!=(const Person& lhs, const Person& rhs) {
        return !(lhs == rhs);
    }
    bool operator>(const Person& lhs, const Person& rhs) {
        return rhs < lhs; // a > b 等价于 b < a
    }
    bool operator<=(const Person& lhs, const Person& rhs) {
        return !(lhs > rhs); // a <= b 等价于 !(b < a)
    }
    bool operator>=(const Person& lhs, const Person& rhs) {
        return !(lhs < rhs); // a >= b 等价于 !(a < b)
    }
    
    int main() {
        Person p1("Alice", 30);
        Person p2("Bob", 25);
        Person p3("Alice", 30);
        Person p4("Charlie", 30);
    
        std::cout << "p1 == p2: " << (p1 == p2) << std::endl; // 0 (false)
        std::cout << "p1 == p3: " << (p1 == p3) << std::endl; // 1 (true)
        std::cout << "p1 < p2: " << (p1 < p2) << std::endl;  // 0 (false) (p1年龄大)
        std::cout << "p2 < p1: " << (p2 < p1) << std::endl;  // 1 (true)
        std::cout << "p1 < p4: " << (p1 < p4) << std::endl;  // 1 (true) (p1姓名A < p4姓名C)
        std::cout << "p4 < p1: " << (p4 < p1) << std::endl;  // 0 (false)
    
        return 0;
    }

    这里需要注意

    const
    正确性,成员函数版本的比较运算符通常应该是
    const
    成员函数,因为它不应该修改对象的状态。

2. 使用 C++20 的

operator<=>
(三路比较 / 飞船运算符)

这是现代C++推荐的做法,它极大地简化了比较运算符的实现。通过一个

operator<=>
,编译器可以自动生成所有六个关系运算符(
==
,
!=
,
<
,
>
,
<=
,
>=
)。

#include 
#include 
#include  // 包含 std::strong_ordering 等

class PersonCpp20 {
public:
    std::string name;
    int age;

    PersonCpp20(std::string n, int a) : name(std::move(n)), age(a) {}

    // 使用 default 实现三路比较
    // 如果类的所有成员都支持 <=>,编译器可以自动生成这个默认实现
    // 否则,我们需要手动实现
    auto operator<=>(const PersonCpp20& other) const = default;

    // 如果需要自定义比较逻辑,可以这样实现:
    /*
    std::strong_ordering operator<=>(const PersonCpp20& other) const {
        if (auto cmp = age <=> other.age; cmp != 0) {
            return cmp; // 年龄不同,直接返回年龄的比较结果
        }
        return name <=> other.name; // 年龄相同,比较姓名
    }
    */

    // 同样,辅助输出
    friend std::ostream& operator<<(std::ostream& os, const PersonCpp20& p) {
        return os << "PersonCpp20(" << p.name << ", " << p.age << ")";
    }
};

int main() {
    PersonCpp20 p1("Alice", 30);
    PersonCpp20 p2("Bob", 25);
    PersonCpp20 p3("Alice", 30);
    PersonCpp20 p4("Charlie", 30);

    std::cout << "p1 == p2: " << (p1 == p2) << std::endl; // 0
    std::cout << "p1 == p3: " << (p1 == p3) << std::endl; // 1
    std::cout << "p1 < p2: " << (p1 < p2) << std::endl;  // 0
    std::cout << "p2 < p1: " << (p2 < p1) << std::endl;  // 1
    std::cout << "p1 < p4: " << (p1 < p4) << std::endl;  // 1
    std::cout << "p4 < p1: " << (p4 < p1) << std::endl;  // 0

    // 甚至可以直接比较三路比较结果
    std::cout << "(p1 <=> p2 == 0): " << (p1 <=> p2 == 0) << std::endl; // 0
    std::cout << "(p1 <=> p3 == 0): " << (p1 <=> p3 == 0) << std::endl; // 1

    return 0;
}

operator<=>
返回一个表示比较结果的枚举类型,如
std::strong_ordering
std::weak_ordering
std::partial_ordering
= default
是其最强大的特性之一,它让编译器根据成员的顺序和它们自身的比较规则自动生成比较逻辑。

为什么我们需要自定义对象比较?

在我看来,自定义对象比较是面向对象编程中不可或缺的一环,它赋予了我们自定义类型以“值语义”的能力。说白了,当你创建了一个

Person
对象,你关心的往往不是它在内存中的地址,而是它所代表的那个“人”是否与另一个“人”在逻辑上是同一个,或者在某种排序规则下,谁先谁后。

默认行为的局限性:C++为我们自定义的类提供的默认比较行为,仅仅是比较对象的内存地址(对于指针或引用),或者执行成员逐一的默认比较(对于结构体或聚合类,如果它们没有自定义比较)。这在大多数情况下都是无意义的。比如,两个

Person
对象即使包含完全相同的姓名和年龄,如果它们是不同的实例,默认的
==
操作符会认为它们不相等,因为它们的内存地址不同。这显然与我们对“相等”的直观理解相悖。

实现抽象与逻辑正确性:通过重载比较运算符,我们能够将对象内部的复杂数据结构抽象成一个简单的比较结果。这不仅让代码更易读、更符合直觉,也确保了业务逻辑的正确性。想象一下,在一个学生管理系统中,如果不能正确比较两个

Student
对象是否是同一个人(例如通过学号),那么很多核心功能,如查找、去重、排序,都将无法正常工作。

与标准库的无缝集成:C++标准库提供了大量强大的容器和算法,如

std::map
std::set
std::sort
std::unique
等。这些工具都高度依赖于对象的比较能力。例如,
std::map
std::set
需要知道如何对键进行排序(默认使用
<
),而
std::sort
也需要一个排序准则。如果没有自定义的比较运算符,这些工具就无法有效地处理我们的自定义类型。这就像你买了一辆跑车,却发现没有方向盘和油门,那它就无法在赛道上驰骋。

所以,自定义比较操作不仅仅是语法糖,它是赋予我们自定义类型以完整生命力,让它们能够融入C++生态系统的关键一步。

如何选择合适的比较策略:成员函数 vs. 非成员函数?

这确实是一个常见的选择困境,尤其是在C++20之前,它关乎到代码的封装性、灵活性以及一些微妙的语言特性。在我看来,这两种方式各有其适用场景,但非成员函数通常更具优势。

1. 成员函数方式

当我们将比较运算符定义为类的成员函数时,它通常长这样:

bool MyClass::operator==(const MyClass& other) const;

ShopEx助理
ShopEx助理

一个类似淘宝助理、ebay助理的客户端程序,用来方便的在本地处理商店数据,并能够在本地商店、网上商店和第三方平台之间实现数据上传下载功能的工具。功能说明如下:1.连接本地商店:您可以使用ShopEx助理连接一个本地安装的商店系统,这样就可以使用助理对本地商店的商品数据进行编辑等操作,并且数据也将存放在本地商店数据库中。默认是选择“本地未安装商店”,本地还未安

下载
  • 优点

    • 直接访问私有成员:这是最明显的优势。如果比较逻辑需要访问类的私有数据,成员函数可以直接访问,无需额外的
      friend
      声明。这在某些情况下简化了代码。
    • 语义自然:从语法上讲,
      obj1 == obj2
      看起来就像
      obj1
      在“询问”它是否与
      obj2
      相等,这与成员函数调用
      obj1.equals(obj2)
      的感觉很相似。
  • 缺点

    • 不对称性:成员函数版本的比较运算符要求左操作数必须是该类的对象(或其派生类)。这意味着
      obj == another_type_obj
      可以工作(如果
      another_type_obj
      可以隐式转换
      MyClass
      ),但
      another_type_obj == obj
      则不行,除非
      another_type_obj
      的类也重载了相应的运算符,或者
      MyClass
      提供了到
      another_type_obj
      的隐式转换。这种不对称性在需要混合类型比较时会造成麻烦。
    • 不适用于左侧隐式转换:如果你的类支持从其他类型进行隐式转换(例如,
      MyString
      可以从
      const char*
      构造),那么
      const char* == myStringObj
      将无法通过成员函数版本的
      operator==
      来调用,因为左侧操作数不是
      MyString
      类型。

2. 非成员函数方式

非成员函数版本的比较运算符通常定义在类的外部,可以声明为

friend
函数,也可以是普通的非
friend
函数。

  • 优点

    • 对称性:这是非成员函数最大的优势。
      operator==(const MyClass& lhs, const MyClass& rhs)
      允许左、右操作数都进行隐式类型转换,使得
      obj == another_type_obj
      another_type_obj == obj
      都能正常工作,只要有合适的转换路径。这对于实现更通用的比较逻辑非常重要。
    • 更好的封装:如果非成员函数不需要访问私有成员,它甚至不需要是
      friend
      。这鼓励我们通过公共接口(getter方法)来获取比较所需的数据,从而提高了类的封装性。
    • 更符合“外部”视角:比较操作,从某种意义上说,是对两个对象之间关系的描述,而不是某个对象自身的行为。将其放在外部,更符合这种“外部视角”。
  • 缺点

    • 需要
      friend
      声明或公共接口
      :如果比较逻辑确实需要访问类的私有成员,那么非成员函数就必须被声明为
      friend
      ,这在一定程度上打破了封装。如果不想使用
      friend
      ,就必须提供公共的getter方法,这有时会暴露不必要的内部细节。

我的建议

在C++20之前,我个人更倾向于非成员非

friend
函数,如果可以的话(即所有比较所需的数据都可以通过公共接口获取)。如果必须访问私有成员,那么非成员
friend
函数
是次优选择,因为它提供了对称性。只有在特殊情况下,例如比较逻辑非常简单且仅涉及本类对象,或者出于性能考虑(尽管现代编译器通常能优化掉这些差异),我才会考虑成员函数。

然而,C++20的

operator<=>
彻底改变了这一格局。它通常作为成员函数实现,但编译器会智能地利用它来合成所有非成员的比较运算符,从而完美地结合了成员函数的直接性和非成员函数的对称性。所以,如果你的项目可以使用C++20,那么
operator<=>
是毫无疑问的首选。

C++20
operator<=>
(三路比较) 的优势与实践

C++20引入的

operator<=>
,也就是我们常说的“飞船运算符”或“三路比较运算符”,在我看来,是C++在处理对象比较方面的一次革命性进步。它不仅仅是语法糖,更是解决了一系列长期存在的痛点,让比较操作变得前所未有的简洁、安全和高效。

核心优势

  1. 减少样板代码 (Boilerplate Reduction):这是最直观的优势。在C++20之前,为了实现完整的六个比较运算符(

    ==
    ,
    !=
    ,
    <
    ,
    >
    ,
    <=
    ,
    >=
    ),你通常需要手动编写至少两个(
    ==
    <
    ),然后通过它们推导出其他四个。这不仅代码量大,而且容易出错。
    operator<=>
    的出现,让你只需实现一个运算符,编译器就能自动合成所有六个!这大大减少了冗余,提升了开发效率。

  2. 保证一致性 (Consistency Guarantee):手动编写多个比较运算符时,很容易出现逻辑不一致的情况。比如,

    a < b
    a > b
    的逻辑可能在不经意间冲突。
    operator<=>
    通过一个单一的比较点来决定所有关系,从根本上杜绝了这种不一致性,确保了所有比较结果的逻辑严谨性。

  3. 默认实现 (Defaulted Implementation):对于那些成员变量本身都支持比较的类(尤其是结构体),你甚至不需要手动编写

    operator<=>
    的实现。只需一行
    auto operator<=>(const MyClass& other) const = default;
    ,编译器就会按照成员声明的顺序,逐个比较成员,并生成正确的比较逻辑。这简直是“懒人福音”,让简单的值类型拥有完整的比较能力变得轻而易举。

  4. 清晰的比较语义 (Clear Comparison Semantics)

    operator<=>
    返回
    std::strong_ordering
    std::weak_ordering
    std::partial_ordering
    这三种类型之一,它们清晰地表达了比较的强度和特性:

    • std::strong_ordering
      :表示强序,等价的值在各个方面都是不可区分的(例如,整数比较)。
    • std::weak_ordering
      :表示弱序,等价的值在排序上是相同的,但在其他方面可能有所不同(例如,大小写不敏感的字符串比较,“Apple”和“apple”等价但可区分)。
    • std::partial_ordering
      :表示偏序,有些值可能无法比较(例如,浮点数的NaN)。 这种明确的类型区分,让开发者能够更好地理解和控制比较行为。

实践应用

  1. 最简单的场景:

    = default
    如果你的类(或结构体)的所有非静态数据成员都支持
    operator<=>
    (例如,基本类型、
    std::string
    、其他自定义的C++20可比较类型),那么你可以直接使用默认实现:

    #include 
    #include  // 必须包含这个头文件
    
    struct Point {
        int x;
        int y;
        auto operator<=>(const Point& other) const = default; // 编译器自动生成
    };
    
    // 现在 Point 对象就可以使用 ==, !=, <, >, <=, >= 进行比较了
    // Point p1{1, 2}, p2{1, 3};
    // p1 < p2 会自动比较 x,然后比较 y

    这在我看来,是C++20最甜的语法糖之一,它让许多简单的数据结构瞬间变得“全能”。

  2. 自定义比较逻辑 当默认的成员逐一比较不符合你的需求时,你需要手动实现

    operator<=>
    。这时,你可以利用
    std::tie
    或者链式比较的模式。

    #include 
    #include 
    #include  // 用于 std::tie
    
    class Product {
    public:
        std::string name;
        double price;
        int id;
    
        Product(std::string n, double p, int i) : name(std::move(n)), price(p), id(i) {}
    
        // 自定义比较逻辑:先按ID,ID相同再按名称,名称相同再按价格
        std::strong_ordering operator<=>(const Product& other) const {
            // 方式一:链式比较 (推荐,更易读)
            if (auto cmp = id <=> other.id; cmp != 0) {
                return cmp;
            }
            if (auto cmp = name <=> other.name; cmp != 0) {
                return cmp;
            }
            return price <=> other.price;
    
            // 方式二:使用 std::tie (简洁,但可能略微牺牲可读性)
            // return std::tie(id, name, price) <=> std::tie(other.id, other.name, other.price);
        }
    
        // 如果只希望 == 运算符默认生成,而其他比较需要自定义,
        // 可以只提供 operator== = default; 然后手动实现 operator<
        // 但有了 <=>

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

318

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1467

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

386

2023.09.04

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

49

2025.11.27

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

PHP WebSocket 实时通信开发
PHP WebSocket 实时通信开发

本专题系统讲解 PHP 在实时通信与长连接场景中的应用实践,涵盖 WebSocket 协议原理、服务端连接管理、消息推送机制、心跳检测、断线重连以及与前端的实时交互实现。通过聊天系统、实时通知等案例,帮助开发者掌握 使用 PHP 构建实时通信与推送服务的完整开发流程,适用于即时消息与高互动性应用场景。

11

2026.01.19

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

相关下载

更多

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.9万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号