运算符重载

简介: 运算符重载


运算符重载概念:对已有的运算符重新进行定义,赋予其另一种功能,以适应不同的数据类型

加号运算符重载

作用:实现两个自定义类型的相加的运算
如果现在我们有一个类如下:

class Person
{
    
public:
    Person()
    {
    
        m_a = 10;
        m_b = 30;
    }
public:
    int m_a;
    int m_b;
};

如果我们想实现

    Person p1;
    Person p2;
    Person p3 = p1 + p2;

按普通的加法肯定是不行,这1就要使用加号运算符重载
1.成员函数重载+号

class Person
{
    
public:
    Person()
    {
    
        m_a = 10;
        m_b = 30;
    }
    //成员函数重载 +
    Person operator+(Person& p)
    {
    
        Person temp;
        temp.m_a = this->m_a + p.m_a;
        temp.m_b = this->m_b + p.m_b;
        return temp;
    }

public:
    int m_a;
    int m_b;
}

本质:

      Person p3 = p1 + p2;
本质:Person p3 = p1.operator+(p2);

2.全局函数重载

Person operator+(Person& p1, Person& p2)
{
    
    Person temp;
    temp.m_a = p1.m_a + p2.m_a;
    temp.m_b = p1.m_b + p2.m_b;
    return temp;
}

本质:

      Person p3 = p1 + p2;
本质:Person p3 = operator+(p1, p2);

运算符重载函数也是可以发生重载的

左移运算符重载

作用:输出自定义类型
比如:
先准备一个类

#include
using namespace std;
class Person
{
    
public:
    Person()
    {
    
        m_a = 10;
        m_b = 10;
    }
    int m_a;
    int m_b;
};

如果我们想实现

Person p;
cout<<p<<endl;

我们就需要利用左移运算符重载

利用全局函数重载左移运算符

ostream& operator<<(ostream &cout, Person p)
{
    
    cout << "m_a= " << p.m_a << "   m_b = " << p.m_b;
    return cout;
}

总结:重载左移运算符配合友元可以实现输出自定义数据类型

递增运算符重载

作用:通过重载递增运算符,实现自己的整型数据

using namespace std;
class Person
{
    
public:
    Person()
    {
    
        m_Num = 0;
    }
    int m_Num;
};

假如我们要实现:

Person p;
p++;
++p;

我们就需要用到递增运算符重载

前置++

using namespace std;
class Person
{
    
public:
    Person()
    {
    
        m_Num = 0;
    }
    //重载前置++运算符  返回引用为了一直对一个数据进行递增操作
    Person& operator++()
    {
    
        m_Num++;
        return *this;
    }
    int m_Num;
};

后置++:

class Person
{
    
    friend ostream& operator<<(ostream& cout, Person p);
public:
    Person()
    {
    
        m_Num = 0;
    }
    //重载后置++运算符   用占位参数来区分后置++
    Person operator++(int)
    {
    
        Person  temp = *this;
        m_Num++;
        return temp;
    }
    int m_Num;
};

赋值运算符重载

C++编译器会有默认的赋值运算符重载,
赋值运算符operator=,对属性进行值拷贝
但这个默认的赋值运算符operator=,只能对属性进行值拷贝也就是浅拷贝,
如果要进行深拷贝的话,就需要自己写这个赋值运算符重载

#include
using namespace std;
class Person
{
    
public:
    Person(int a)
    {
    
        age = new int(a);
    }
    Person& operator=(Person& p)  //赋值运算符重载
    {
    
        if (age != NULL)
        {
    
            delete age;
            age = NULL;
        }
        age = new int(*p.age);
        return *this;  //返回这个时赋值时可以进行连等操作
    }
    ~Person()    //析构函数,在程序结束后释放空间
    {
    
        if (age != NULL)
        {
    
            delete age;
            age = NULL;
        }
    }
public:
    int* age;

};

关系运算符重载

作用:重载关系运算符,可以让两个自定义类型的对象进行对比操作
比如:
比较两个类的大小关系
要实现这个就需要重载关系运算符

#include
#include
using namespace std;
class Person
{
    
public:
    bool operator==(Person& p)
    {
    
        if (this->age == p.age && this->name == p.name)
            return true;
        else
            return false;
    }
    bool operator!=(Person& p)
    {
    
        if (this->age != p.age || this->name == p.name)
            return true;
        else
            return false;
    }
public:
    string name;
    int age;
};

这样我们就可以进行下面操作了

int main()
{
    
    Person p1;
    p1.age = 18;
    p1.name = "abd";
    Person p2;
    p2.age = 18;
    p2.name = "abd";
    if (p1 == p2)
        cout << "yes" << endl;
    if (p1 != p2)
        cout << "yes" << endl;
}

函数调用运算符重载

函数调用运算符()也可以进行重载
由于重载后的使用方法非常像函数的调用,因此称为仿函数
仿函数没有固定写法,非常灵活。

如果我们要写一个打印类

#include
#include
class Mprint
{
    
public:
    void operator()(string s)
    {
    
        cout << s << endl;
    }
};
int main()
{
    
    Mprint mprintf;
    mprintf("hello word");   //由于使用方法特别像函数调用,所以被称为仿函数
}

如果想实现一个加法类:

#include
#include
class MyAdd
{
    
public:
    int operator()(int a, int b)
    {
    
        return a + b;
    }
};
int main()
{
    
    MyAdd add;
    int ret = add(20, 30);
    cout << ret << endl;
    //匿名对象调用
    cout << MyAdd()(20, 30) << endl;
}
相关文章
|
边缘计算 编译器 数据中心
X86架构与Arm架构的主要区别分析
X86架构与Arm架构的主要区别分析
1575 0
|
8月前
|
机器学习/深度学习 人工智能 编解码
Text to Bark:让狗狗听懂人话!全球首个AI"狗语"生成器,137种狗狗口音任君挑选
ElevenLabs推出的Text to Bark是全球首个能将文本转换为逼真狗吠声的AI模型,支持多种犬种选择并适配智能家居设备,其核心技术基于深度神经网络训练。
1569 15
Text to Bark:让狗狗听懂人话!全球首个AI"狗语"生成器,137种狗狗口音任君挑选
|
前端开发 开发工具 Android开发
跨平台开发工具
跨平台开发工具
598 6
|
10月前
|
存储 人工智能 文字识别
MME-CoT:多模态模型推理能力终极评测!六大领域细粒度评估,港中大等机构联合推出
MME-CoT 是由港中文等机构推出的用于评估大型多模态模型链式思维推理能力的基准测试框架,涵盖数学、科学、OCR、逻辑、时空和一般场景等六个领域,提供细粒度的推理质量、鲁棒性和效率评估。
625 0
|
数据采集 人工智能 数据挖掘
【钉钉杯大学生大数据挑战赛】初赛B 航班数据分析与预测 Python代码实现Baseline
本文提供了参加"钉钉杯大学生大数据挑战赛"初赛B的航班数据分析与预测项目的Python代码实现Baseline。内容包括题目背景、思路分析、训练集和测试集的预处理、模型训练与预测、特征重要性分析,以及代码下载链接。预处理步骤涉及读取数据、时间信息处理、前序航班延误时间计算、天气信息匹配等。模型训练使用了Gradient Boosting Classifier,并对模型的准确率和特征重要性进行了评估。
573 0
|
存储 开发框架 搜索推荐
【Uniapp 专栏】Uniapp 打造的音乐类应用案例剖析
【5月更文挑战第12天】使用Uniapp跨平台框架开发的音乐应用,提供丰富音乐库和个性化推荐,用户可轻松搜索、创建歌单及离线收听。应用采用先进音频技术确保流畅播放,设计简洁美观,集成社交分享功能。通过优化搜索和推荐系统,提升用户体验。此案例展现了Uniapp在音乐应用开发的优势和潜力,为开发者带来启示。
375 2
|
存储 人工智能 弹性计算
通义万相AI绘画创作评测及图文搭建教程
【7月更文挑战第4天】阿里云的通义万相是AI绘画模型,结合ECS、OSS和API服务,提供无缝创作环境。用户上传图片至OSS,模型通过签名URL下载图片,然后生成AI艺术作品。模型服务具有高性能、易集成的特点,适用于多种场景如设计、广告等。用户可按指示在阿里云官网注册、充值、开通服务并部署。项目评测显示,其集成便捷、响应快、泛化能力强,但仍有改进空间,如增加图像控制选项和批量处理能力。相对于竞品,通义万相在成本、易用性和应用场景上有竞争力,值得推荐。
12215 9
|
SQL 关系型数据库 MySQL
MySQL服务的状态如何查看?
【5月更文挑战第23天】MySQL服务的状态如何查看?
4263 1
|
传感器 芯片
嵌入式通信协议全解析:SPI、I²C、UART详解(附带面试题)
通信是指人与人或人与自然之间通过某种行为或媒介进行的信息交流与传递。从广义上来说,通信是指需要信息的双方或多方在不违背各自意愿的情况下采用任意方法、任意媒质,将信息从某方准确安全地传送到另方。在出现电波传递通信后,通信被单一解释为信息的传递,是指由一地向另一地进行信息的传输与交换,其目的是传输消息。通信方式包括利用“电”来传递消息的电信,这种通信具有迅速、准确、可靠等特点,且几乎不受时间、地点、空间、距离的限制,因而得到了飞速发展和广泛应用。
5029 0