C++之operator

1、operator

        operator是C++关键字之一, 称为重载运算符(也称“重载操作符”)所谓重载,就是赋予新的含义。函数重载(Function Overloading)可以让一个函数名有多种功能,在不同情况下进行不同的操作。运算符重载(Operator Overloading)也是一个道理,同一个运算符可以有不同的功能

        例如,+号可以对不同类型(int、float 等)的数据进行加法操作;<<既是位移运算符,又可以配合cout向控制台输出数据。C++本身已经对这些运算符进行了重载。C++也允许程序员自己重载运算符,这给我们带来了很大的便利。

实现运算符重载的方式通常有以下两种:

        (1)运算符重载实现为类的成员函数;

        (2)运算符重载实现为非类的成员函数(即全局函数)。

2、重载为类的成员函数(以“==”举例)

        在类体中声明(定义)需要重载的运算符,声明方式跟普通的成员函数一样,只不过运算符重载函数的名字是【operator紧跟一个C++预定义的操作符】【以“==”为例,函数名就是“operator==”

(1)在类中实现函数的逻辑:


#include <iostream>
using namespace std;class student
{
private:char name[10];int age;
public:student(int nAge)  // 构造函数,对私有变量赋值{this->age = nAge;}bool operator==(const student& ps)   // 重载了“==”运算符,用于判断两个值是否相等{if (this->age == ps.age)return true;return false;}
};int main()
{student p1(10);student p2(10);if (p1 == p2)cout << "p1 is equal with p2." << endl;elsecout << "p1 is not equal with p2." << endl;return 0;
}

输出:

p1 is equal with p2.

(2)在类外实现函数的逻辑:


#include <iostream>
using namespace std;class student
{
private:char name[10];int age;
public:student(int nAge)  // 构造函数,对私有变量赋值{this->age = nAge;}bool operator==(const student& ps);   // 重载了“==”运算符,用于判断两个值是否相等
};bool student::operator==(const student& ps)
{if (this->age == ps.age)return true;return false;
}int main()
{student p1(10);student p2(10);if (p1 == p2)cout << "p1 is equal with p2." << endl;elsecout << "p1 is not equal with p2." << endl;return 0;
}

输出:

p1 is equal with p2.

3、重载为全局函数(以“==”举例)

        对于全局重载运算符,代表左操作数的参数必须被显式指定。

#include <iostream>
using namespace std;class student
{
public:char name[10];int age;
};// 左操作数的类型必须被显式指定;此处指定的类型为person类
bool operator==(student const& p1, student const& p2)
{if (p1.age == p2.age)return true;elsereturn false;
}int main()
{student p1;student p2;p1.age = 18;p2.age = 18;if (p1 == p2)cout << "p1 is equal with p2." << endl;elsecout << "p1 is NOT equal with p2." << endl;return 0;
}

输出:

p1 is equal with p2.

4、重载“数学运算符”

        四则运算符(+、-、*、/、+=、-=、*=、/=)和关系运算符(>、<、<=、>=、==、!=)都是数学运算符,它们在实际开发中非常常见,被重载的几率也很高,并且有着相似的重载格式。本文以复数类 Complex 为例对它们进行重载,重在演示运算符重载的语法以及规范。

        复数能够进行完整的四则运算,但不能进行完整的关系运算:我们只能判断两个复数是否相等,但不能比较它们的大小,所以不能对 >、<、<=、>= 进行重载

#include <iostream>
#include <cmath>
using namespace std;//复数类
class Complex {
public://构造函数Complex(double real = 0.0, double imag = 0.0) : m_real(real), m_imag(imag) { }
public://运算符重载//以全局函数的形式重载friend Complex operator+(const Complex& c1, const Complex& c2);friend Complex operator-(const Complex& c1, const Complex& c2);friend Complex operator*(const Complex& c1, const Complex& c2);friend Complex operator/(const Complex& c1, const Complex& c2);friend bool operator==(const Complex& c1, const Complex& c2);friend bool operator!=(const Complex& c1, const Complex& c2);//以成员函数的形式重载Complex& operator+=(const Complex& c);Complex& operator-=(const Complex& c);Complex& operator*=(const Complex& c);Complex& operator/=(const Complex& c);
public://成员函数double real() const { return m_real; }double imag() const { return m_imag; }
private:double m_real;  //实部double m_imag;  //虚部
};//重载为全局函数
//重载+运算符
Complex operator+(const Complex& c1, const Complex& c2) {Complex c;c.m_real = c1.m_real + c2.m_real;c.m_imag = c1.m_imag + c2.m_imag;return c;
}
//重载-运算符
Complex operator-(const Complex& c1, const Complex& c2) {Complex c;c.m_real = c1.m_real - c2.m_real;c.m_imag = c1.m_imag - c2.m_imag;return c;
}
//重载*运算符  (a+bi) * (c+di) = (ac-bd) + (bc+ad)i
Complex operator*(const Complex& c1, const Complex& c2) {Complex c;c.m_real = c1.m_real * c2.m_real - c1.m_imag * c2.m_imag;c.m_imag = c1.m_imag * c2.m_real + c1.m_real * c2.m_imag;return c;
}
//重载/运算符  (a+bi) / (c+di) = [(ac+bd) / (c²+d²)] + [(bc-ad) / (c²+d²)]i
Complex operator/(const Complex& c1, const Complex& c2) {Complex c;c.m_real = (c1.m_real * c2.m_real + c1.m_imag * c2.m_imag) / (pow(c2.m_real, 2) + pow(c2.m_imag, 2));c.m_imag = (c1.m_imag * c2.m_real - c1.m_real * c2.m_imag) / (pow(c2.m_real, 2) + pow(c2.m_imag, 2));return c;
}
//重载==运算符
bool operator==(const Complex& c1, const Complex& c2) {if (c1.m_real == c2.m_real && c1.m_imag == c2.m_imag) {return true;}else {return false;}
}
//重载!=运算符
bool operator!=(const Complex& c1, const Complex& c2) {if (c1.m_real != c2.m_real || c1.m_imag != c2.m_imag) {return true;}else {return false;}
}//重载为成员函数
//重载+=运算符
Complex& Complex::operator+=(const Complex& c) {this->m_real += c.m_real;this->m_imag += c.m_imag;return *this;
}
//重载-=运算符
Complex& Complex::operator-=(const Complex& c) {this->m_real -= c.m_real;this->m_imag -= c.m_imag;return *this;
}
//重载*=运算符
Complex& Complex::operator*=(const Complex& c) {this->m_real = this->m_real * c.m_real - this->m_imag * c.m_imag;this->m_imag = this->m_imag * c.m_real + this->m_real * c.m_imag;return *this;
}
//重载/=运算符
Complex& Complex::operator/=(const Complex& c) {this->m_real = (this->m_real * c.m_real + this->m_imag * c.m_imag) / (pow(c.m_real, 2) + pow(c.m_imag, 2));this->m_imag = (this->m_imag * c.m_real - this->m_real * c.m_imag) / (pow(c.m_real, 2) + pow(c.m_imag, 2));return *this;
}int main() {Complex c1(25, 35);Complex c2(10, 20);Complex c3(1, 2);Complex c4(4, 9);Complex c5(34, 6);Complex c6(80, 90);Complex c7 = c1 + c2;Complex c8 = c1 - c2;Complex c9 = c1 * c2;Complex c10 = c1 / c2;cout << "c7 = " << c7.real() << " + " << c7.imag() << "i" << endl;cout << "c8 = " << c8.real() << " + " << c8.imag() << "i" << endl;cout << "c9 = " << c9.real() << " + " << c9.imag() << "i" << endl;cout << "c10 = " << c10.real() << " + " << c10.imag() << "i" << endl;c3 += c1;c4 -= c2;c5 *= c2;c6 /= c2;cout << "c3 = " << c3.real() << " + " << c3.imag() << "i" << endl;cout << "c4 = " << c4.real() << " + " << c4.imag() << "i" << endl;cout << "c5 = " << c5.real() << " + " << c5.imag() << "i" << endl;cout << "c6 = " << c6.real() << " + " << c6.imag() << "i" << endl;if (c1 == c2) {cout << "c1 == c2" << endl;}if (c1 != c2) {cout << "c1 != c2" << endl;}return 0;
}

输出:

c7 = 35 + 55i
c8 = 15 + 15i
c9 = -450 + 850i
c10 = 1.9 + -0.3i
c3 = 26 + 37i
c4 = -6 + -11i
c5 = 220 + 4460i
c6 = 5.2 + 1.592i
c1 != c2

5、重载“>>”和“<<”

        在C++中,标准库本身已经对左移运算符“<<”和右移运算符“>>”分别进行了重载,使其能够用于不同数据的输入输出,但是输入输出的对象只能是C++内置的数据类型(例如 bool、int、double 等)和标准库所包含的类类型(例如 string、complex、ofstream、ifstream 等)。

        如果我们自己定义了一种新的数据类型,需要用输入输出运算符去处理,那么就必须对它们进行重载。本节以前面的complex类为例来演示输入输出运算符的重载。

        本节要达到的目标是让复数的输入输出和int、float等基本类型一样简单。假设num1、num2是复数,那么输出形式就是:

        cout<<num1<<num2<<endl;

输入形式就是:

        cin>>num1>>num2;

        cout是ostream类的对象,cin是istream类的对象,要想达到这个目标,就必须以全局函数(友元函数)的形式重载“<<”和“>>”。

#include <iostream>
using namespace std;class complex {
public:complex(double real = 0.0, double imag = 0.0) : m_real(real), m_imag(imag) { };
public:friend complex operator+(const complex& A, const complex& B);friend complex operator-(const complex& A, const complex& B);friend complex operator*(const complex& A, const complex& B);friend complex operator/(const complex& A, const complex& B);//重载输入输出运算符friend istream& operator>>(istream& in, complex& A);friend ostream& operator<<(ostream& out, complex& A);
private:double m_real;  //实部double m_imag;  //虚部
};//重载加法运算符
complex operator+(const complex& A, const complex& B) {complex C;C.m_real = A.m_real + B.m_real;C.m_imag = A.m_imag + B.m_imag;return C;
}//重载减法运算符
complex operator-(const complex& A, const complex& B) {complex C;C.m_real = A.m_real - B.m_real;C.m_imag = A.m_imag - B.m_imag;return C;
}//重载乘法运算符
complex operator*(const complex& A, const complex& B) {complex C;C.m_real = A.m_real * B.m_real - A.m_imag * B.m_imag;C.m_imag = A.m_imag * B.m_real + A.m_real * B.m_imag;return C;
}//重载除法运算符
complex operator/(const complex& A, const complex& B) {complex C;double square = A.m_real * A.m_real + A.m_imag * A.m_imag;C.m_real = (A.m_real * B.m_real + A.m_imag * B.m_imag) / square;C.m_imag = (A.m_imag * B.m_real - A.m_real * B.m_imag) / square;return C;
}//重载输入运算符
istream& operator>>(istream& in, complex& A) {in >> A.m_real >> A.m_imag;return in;
}//重载输出运算符
ostream& operator<<(ostream& out, complex& A) {out << A.m_real << " + " << A.m_imag << " i ";;return out;
}int main() {complex c1, c2, c3;cin >> c1 >> c2;c3 = c1 + c2;cout << "c1 + c2 = " << c3 << endl;c3 = c1 - c2;cout << "c1 - c2 = " << c3 << endl;c3 = c1 * c2;cout << "c1 * c2 = " << c3 << endl;c3 = c1 / c2;cout << "c1 / c2 = " << c3 << endl;return 0;
}

输出:

1.2 3.4
56.4 7.8
c1 + c2 = 57.6 + 11.2 i
c1 - c2 = -55.2 + -4.4 i
c1 * c2 = 41.16 + 201.12 i
c1 / c2 = 7.24615 + 14.0308 i

6、其他重载

7、参考

        参考案例过多,暂不进行列举。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/6182.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于Unity+Vue通信交互的WebGL项目实践

unity-webgl 是无法直接向vue项目进行通信的&#xff0c;需要一个中间者 jslib 文件 jslib当作中间者&#xff0c;unity与它通信&#xff0c;前端也与它通信&#xff0c;在此基础上三者之间进行了通信对接 看过很多例子&#xff1a;介绍的都不是很详细&#xff0c;不如自己写&…

【MATLAB源码-第199期】基于MATLAB的深度学习(CNN)数字、模拟调制识别仿真,输出识别率。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 基于深度学习的调制识别系统利用复杂的数学模型和算法来识别和分类从不同来源接收到的无线信号的调制类型。这种技术的应用广泛&#xff0c;特别是在无线通信、电子战、频谱监测和认知无线电等领域中具有重要价值。调制识别系…

算法导论 总结索引 | 第三部分 第十二章:二叉搜索树

1、搜索树数据结构 支持 许多动态集合操作&#xff0c;包括 SEARCH、MINIMUM、MAXIMUM、PREDECESSOR、SUCCESSOR、INSERT 和 DELETE 等。使用搜索树 既可以作为一个字典 又可以作为一个优先队列 2、二叉搜索树上的基本操作 所花费的时间 与这棵树的高度成正比。对于有n个结点的…

汽车热辐射、热传导、热对流模拟加速老化太阳光模拟器系统

汽车整车结构复杂&#xff0c;材料种类繁多&#xff0c;在使用过程中会面临各种严酷气候环境的考验&#xff0c;不可避免会出现零部件材料老化、腐蚀等不良现象&#xff0c;从而影响汽车的外观、功能&#xff0c;甚至产生安全隐患。因此&#xff0c;分析汽车零部件材料老化腐蚀…

深入剖析Tomcat(五) 剖析Servlet容器并实现一个简易Context与Wrapper容器

上一章介绍了Tomcat的默认连接器&#xff0c;后续程序都会使用默认连接器。前面有讲过Catalina容器的两大块内容就是连接器与Servlet容器。不同于第二章的自定义丐版Servlet容器&#xff0c;这一章就来探讨下Catalina中的真正的Servlet容器究竟长啥样。 四种容器 在Catalina中…

音视频入门基础:像素格式专题(1)——RGB简介

一、像素格式简介 像素格式&#xff08;pixel format&#xff09;指像素色彩按分量的大小和排列。这种格式以每个像素所使用的总位数以及用于存储像素色彩的红、绿、蓝和 alpha 分量的位数指定。在音视频领域&#xff0c;常用的像素格式包括RGB格式和YUV格式&#xff0c;本文…

mysql添加远程登录账户

为了远程连接&#xff0c;您必须使MySQL将端口3306绑定到my.cnf中计算机的IP地址。然后&#xff0c;您必须同时在localhost和&#xff05;通配符中创建用户&#xff0c;并在所有DB上授予权限。 修改my.cnf&#xff0c;如果不存在这行则添加&#xff0c;可以输入0.0.0.0 bind-ad…

Java面试题:并发编程中,什么是死锁,如何避免它?

什么是死锁&#xff1f; 死锁是指在多线程环境中&#xff0c;由于资源分配不当或进程运行顺序不合理&#xff0c;导致涉及的线程都处于等待状态&#xff0c;无法继续执行的一种状况。在Java中&#xff0c;死锁通常发生在多个线程互相等待对方持有的锁&#xff0c;但没有任何一…

【海博】雅思该怎么练?

文章目录 前言 备考计划 模拟考试 参考资料 前言 见《【海博】浅析海博深造》 见《【海博】雅思和托福该考哪个&#xff1f;》 见《【海博】雅思该怎么考&#xff1f;》 见《【海博】雅思考什么&#xff1f;》 备考计划 第一周确定你的目标考试分数。 做一套雅思模拟试题&…

npm install报错

总结&#xff1a;没有安装visual studio 2017以上带有C桌面开发的版本 #开始试错 #报错总信息mingw_x64_win版本 百度网盘链接: link 提取码&#xff1a;3uou #尝试用mingw配置个C编译器&#xff0c;并配置环境变量&#xff0c;失败 #只认可使用VS!GIthub原址: 链接: link 上…

python公务用车医院校园企业车辆管理系统

本 Python版本&#xff1a;python3.7 前端&#xff1a;vue.jselementui 框架&#xff1a;django/flask都有,都支持 后端&#xff1a;python 数据库&#xff1a;mysql 数据库工具&#xff1a;Navicat 开发软件&#xff1a;PyCharm 公务用车管理智慧云服务监管平台有管理员和用户…

java中的字节流和File类

目录 正文&#xff1a; 1.File类 1.1概述 1.2常用方法 2.FileInputStream 2.1概述 2.2常用方法 3.FileOutputStream 3.1概述 3.2常用方法 总结&#xff1a; 正文&#xff1a; 1.File类 1.1概述 File类是Java中用来表示文件或目录的类&#xff0c;它提供了一系列方…

【项目学习01_2024.05.02_Day04】

学习笔记 4 课程分类查询4.1需求分析4.2 接口定义4.3 接口开发4.3.1 树型表查询4.3.2 开发Mapper 4 课程分类查询 4.1需求分析 有课程分类的需求 course_category课程分类表的结构 这张表是一个树型结构&#xff0c;通过父结点id将各元素组成一个树。 利用mybatis-plus-gen…

Sass语法---sass的安装和引用

什么是Sass Sass&#xff08;英文全称&#xff1a;Syntactically Awesome Stylesheets&#xff09; Sass 是一个 CSS 预处理器。 Sass 是 CSS 扩展语言&#xff0c;可以帮助我们减少 CSS 重复的代码&#xff0c;节省开发时间。 Sass 完全兼容所有版本的 CSS。 Sass 扩展了…

centos7安装真的Redmine-5.1.2+ruby-3.0.0

下载redmine-5.1.2.tar.gz&#xff0c;上传到/usr/local/目录下 cd /usr/local/ tar -zxf redmine-5.1.2.tar.gz cd redmine-5.1.2 cp config/database.yml.example config/database.yml 配置数据连接 #编辑配置文件 vi config/database.yml #修改后的内容如下 product…

【介绍下Apache的安装与目录结构】

&#x1f3a5;博主&#xff1a;程序员不想YY啊 &#x1f4ab;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f917;点赞&#x1f388;收藏⭐再看&#x1f4ab;养成习惯 ✨希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出…

使用python遍历各个文件夹下以docx为后缀的文件并移动到指定的文件夹

使用python遍历各个文件夹下以docx为后缀的文件并移动到指定的文件夹 import os import shutilsource_directory "./" # 源文件夹路径 destination_directory "../word" # 目标文件夹路径 file_extension ".docx" # 文件后缀名# 遍历源文…

Vitis HLS 学习笔记--MAXI手动控制突发传输

目录 1. 简介 2. MAXI 突发传输详解 2.1 突发传输的前置条件 2.2 hls::burst_maxi 详解 2.2.1 基本知识 2.2.2 hls::burst_maxi 构造函数 2.2.3 hls::burst_maxi 读取方法 2.2.4 hls::burst_maxi 写入方法 2.3 示例一 2.4 示例二 3. 总结 1. 简介 这篇文章探讨了在…

深入剖析序列的特殊方法

协议和鸭子类型 开始之前,我先介绍协议和鸭子类型。在Python中创建功能完善的序列类型无需继承,实现符合序列协议的方法即可。那么说的协议是什么呢? 在面向对象编程中,协议是非正式的接口,只在文档中定义,不在代码中定义。例如:Python的序列协议只要实现__len__和__g…

Python版本管理工具-pyenv

Pyenv是一个Python版本管理工具。 Pyenv允许用户在同一台机器上安装多个版本的Python&#xff0c;并能够轻松切换使用这些版本。 一、安装 Mac下直接使用Homebrew安装 # 更新 Homebrew 的软件列表 brew update # 安装pyenv brew install pyenv# 验证是否安装成功 pyenv -v# …