广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C语言运算符的重载详解
  • 478
分享到

C语言运算符的重载详解

2024-04-02 19:04:59 478人浏览 八月长安
摘要

目录运算符重载的规则 1. 运算符重载为成员函数2. 运算符重载为非成员函数        

运算符重载的规则 

  • 思考:用“+”、“-”能够实现复数的加减运算吗?
  • 实现复数加减运算的方法:

             ——重载“+”、“-”运算符

  • 运算符重载是对已有的运算符赋予多重含义,使同一个运算符作用于不同类型的数据时导致不同的行为。
  • c++ 几乎可以重载全部的运算符,而且只能够重载C++中已经有的。

                不能重载的运算符:“.”、“.*”、“::”、“?:”

  • 重载之后运算符的优先级和结合性都不会改变。
  • 运算符重载是针对新类型数据的实际需要,对原有运算符进行适当的改造。例如:                 

                使复数类的对象可以用“+”运算符实现加法;

                使时钟类对象可以用“++”运算符实现时间增加1秒。

  • 重载为类的非静态成员函数;
  • 重载为非成员函数。 

1. 运算符重载为成员函数

重载为类成员的运算符函数定义形式:

函数类型 operator 运算符(形参)

{

......

}

参数个数=原操作数个数-1 (后置++、--除外) 

双目运算符重载规则 

  • 如果要重载 B 为类成员函数,使之能够实现表达式 oprd1 B oprd2,其中 oprd1 为A 类对象,则 B 应被重载为 A 类的成员函数,形参类型应该是 oprd2 所属的类型。
  • 经重载后,表达式 oprd1 B oprd2 相当于 oprd1.operator B(oprd2) 。 

例1 复数类加减法运算重载为成员函数

要求:将+、-运算重载为复数类的成员函数。

规则:实部和虚部分别相加减。

操作数:两个操作数都是复数类的对象。

#include<iOStream>
using namespace std;
//定义复数类
class Complex
{ 
	public:
		Complex(double x=0.0, double y=0.0);  //构造函数
		Complex(const Complex &c);  //复制构造函数 
		~Complex();  //析构函数
		void display() const;  //复数打印函数
		Complex operator +(const Complex &c) const;  //加法运算符重载
		Complex operator -(const Complex &c) const;  //减法运算符重载 
	private:
		double real;  //实部
		double imag;  //虚部 
}; 
Complex::Complex(double x, double y): real(x), imag(y){}
Complex::~Complex(){}
Complex::Complex(const Complex &c): real(c.real), imag(c.imag){}
void Complex::display() const
{
	cout<<"("<<real<<", "<<imag<<")"<<endl;	
}
Complex Complex::operator +(const Complex &c) const
{
	Complex temp;
	temp.real = real + c.real;
	temp.imag = imag + c.imag;
	return temp;
	
	//return Complex(real+c.real, imag+c.imag); 
}
Complex Complex::operator -(const Complex &c) const
{
	Complex temp;
	temp.real = real - c.real;
	temp.imag = imag - c.imag;
	return temp;
}
int main()
{
	Complex c1(6.7, 4.4);
	Complex c2(3.2, 1.7);
	cout<<"c1 = ";
	c1.display();
	cout<<endl;
	cout<<"c2 = ";
	c2.display();
	cout<<endl;
	Complex c3, c4;
	c3 = c1 + c2;
	c4 = c1 - c2;
	cout<<"c1 + c2 = ";
	c3.display();
	cout<<endl;
	cout<<"c1 - c2 = ";
	c4.display();
	cout<<endl;
	return 0;
} 

运行结果:

c1 = (6.7, 4.4)
c2 = (3.2, 1.7)
c1 + c2 = (9.9, 6.1)
c1 - c2 = (3.5, 2.7)

前置单目运算符++和--重载规则 

  • 如果要重载 U 为类成员函数,使之能够实现表达式 U oprd,其中 oprd 为A类对象,则 U 应被重载为 A 类的成员函数,无形参。
  • 经重载后,表达式 U oprd 相当于 oprd.operator U()。 

后置单目运算符++和--重载规则 

  • 如果要重载 ++或--为类成员函数,使之能够实现表达式 oprd++ 或 oprd-- ,其中 oprd 为A类对象,则 ++或-- 应被重载为 A 类的成员函数,且具有一个 int 类型形参。
  • 经重载后,表达式 oprd++ 相当于 oprd.operator ++(0)。 

例2 重载前置++和后置++为时钟类成员函数

前置单目运算符,重载函数没有形参。         

后置++运算符,重载函数需要有一个int形参。         

操作数是时钟类的对象。         

实现时间增加1秒钟。

#include<iostream>
using namespace std;
class Clock
{
	public:
		Clock(int hour=0, int minute=0, int second=0);
		~Clock();
		void showTime() const;
		Clock& operator ++();  //前置单目运算符重载
		Clock operator ++(int);  //后置单目运算符重载 
	private:
		int hour;
		int minute;
		int second;
};
Clock::Clock(int hour, int minute, int second)
{
	if(0<=hour && hour<24 && 0<=minute && minute<60 && 0<=second && second<60)
	{
		this->hour = hour;
		this->minute = minute;
		this->second = second;
	}
	else
	{
		cout<<"Time error!"<<endl;
	}
}
Clock::~Clock(){}
void Clock::showTime() const
{
	cout<<hour<<":"<<minute<<":"<<second<<endl;
}
Clock& Clock::operator ++()  //函数返回值是对象的引用是为了更高效(减少临时对象的生成)
{
	second++;
	if(second >= 60)
	{
		second %= 60;
		minute++;
		if(minute >= 60)
		{
			minute %= 60;
			hour = (hour + 1) % 24;
		}
	}
	return *this;
}
Clock Clock::operator ++(int)  //注意形参表中的整型参数
{
	Clock old = *this;
	++(*this);   //调用前置“++”运算符
	
	return old; 
} 
int main()
{
	Clock myClock(23, 59, 59); 
 	cout << "First time output: "; 
 	myClock.showTime(); 
 	cout << "Show myClock++: "; 
 	(myClock++).showTime(); 
 	cout << "Show ++myClock: "; 
 	(++myClock).showTime();
	return 0;
}

运行结果:

First time output: 23:59:59
Show myClock++: 23:59:59
Show ++myClock: 0:0:1

2. 运算符重载为非成员函数          

有些运算符不能重载为成员函数,例如二元运算符的左操作数不是对象,或者是不能由我们重载运算符的对象。 

运算符重载为非成员函数的规则 

  • 函数的形参代表依自左至右次序排列的各操作数。
  • 重载为非成员函数时:

                参数个数=原操作数个数(后置++、--除外)。

  • 至少应该有一个自定义类型的参数。
  • 后置单目运算符++和--的重载函数,形参列表中要增加一个int,但不必写形参名。
  • 如果在运算符的重载函数中需要操作某类对象的私有成员,可以将此函数声明为该类的友元
  • 双目运算符B重载后, 表达式oprd1 B oprd2等同于operator B(oprd1,oprd2)。
  • 前置单目运算符B重载后,表达式B oprd等同于operator B(oprd)。
  • 后置单目运算符++和--重载后,表达式oprd B等同于operator B(oprd,0)。

例3 重载 Complex 的加减法和“<<”运算符为非成员函数

  • 将+、‐(双目)重载为非成员函数,并将其声明为复数类的友元,两个操作数都是复数类的常引用(使用常引用,既保证程序执行的高效率,又保护数据不被随意篡改)。
  • 将<<(双目)重载为非成员函数,并将其声明为复数类的友元,它的左操作数是std::ostream引用,右操作数为复数类的常引用,返回std::ostream引用,用以支持下面形式的输出:        

 cout << a << b;

该输出调用的是:         

operator << (operator << (cout, a), b);

#include<iostream>
using namespace std;
class Complex
{
	public:
		Complex(double x=0.0, double y=0.0);
		~Complex();
		friend Complex operator +(const Complex &c1, const Complex &c2);  //声明为类的友元
		friend Complex operator -(const Complex &c1, const Complex &c2);
		friend ostream& operator <<(ostream &out, const Complex &c);
	private:
		double real;
		double imag;
};
Complex::Complex(double x, double y): real(x), imag(y){}
Complex::~Complex(){}
Complex operator +(const Complex &c1, const Complex &c2)
{
	Complex temp;
	temp.real = c1.real + c2.real;
	temp.imag = c1.imag + c2.imag; 
	return temp;
	
	//return Complex(c1.real+c2.real, c1.imag+c2.imag); 
} 
Complex operator -(const Complex &c1, const Complex &c2)
{
	return Complex(c1.real-c2.real, c1.imag-c2.imag);
}
ostream& operator <<(ostream& out, const Complex &c)
{
	out<<"("<<c.real<<", "<<c.imag<<")";
	return out;
}


int main()
{
	Complex c1(5, 4), c2(2, 10), c3; 
 	cout << "c1 = " << c1 << endl; 
 	cout << "c2 = " << c2 << endl; 
 	c3 = c1 - c2;   //使用重载运算符完成复数减法 
 	cout << "c3 = c1 - c2 = " << c3 << endl; 
 	c3 = c1 + c2;   //使用重载运算符完成复数加法 
 	cout << "c3 = c1 + c2 = " << c3 << endl;
	return 0;
}

运行结果:

c1 = (5, 4)
c2 = (2, 10)
c3 = c1 - c2 = (3, -6)
c3 = c1 + c2 = (7, 14)

类的友元函数重载插入运算符和重载提取运算符在调用时只需像平常cin >> xxx 或 cout << xxx即可,而类的成员函数重载插入运算符和重载提取运算符在调用时是要xxx >> cin 或 xxx << cout,刚好和平常的用法是相反的。而xxx >> cin相当于xxx.operator>>(cin);xxx << cout相当于xxx.operator<<(cout)。

用引用&,减少对象拷贝,增加速度和效率。插入和提取运算符,操作的对象是ostream和istream。如果不加&的话,程序也是没问题的,可以运行。但是在每次调用<<操作的时候,都会产生一个新的临时的ostream对象。包括对“=”运算符的重载也是一样。“=”更能说明这个问题。一般来说是 xxx & opreator = (xxx a);就是因为我们希望对于a=b; 把b的值直接赋值给a就好了,用xxx &做返回类型,返回的是a的别名/引用,不会创建新的临时对象。如果没有&,那么会创建新的临时值返回,低效。重点就是对引用&的使用的理解。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注编程网的更多内容! 

--结束END--

本文标题: C语言运算符的重载详解

本文链接: https://www.lsjlt.com/news/140299.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • C语言运算符的重载详解
    目录运算符重载的规则 1. 运算符重载为成员函数2. 运算符重载为非成员函数        ...
    99+
    2022-11-13
  • C++中的运算符重载详解
    目录1、引例2、类中自动建立的函数3、重载赋值运算符解析总结1、引例 class Complex { private: double Real,Image; public: ...
    99+
    2022-11-13
  • C++运算符重载图文详解
    目录1. 运算符重载1.1 运算符重载为普通函数1.2 运算符重载为成员函数2. 赋值运算符=的重载2.1浅复制与深复制2.2返回值的讨论3. 动态可变长度数组总结1. 运算符重载 ...
    99+
    2022-11-12
  • C++重载的奥义之运算符重载详解
    目录0、引言1、运算符重载定义2、作为成员函数进行重载3、作为全局函数进行重载4、运算符重载的一些规则0、引言 重载,顾名思义从字面上理解就是重复装载,打一个不恰当的比方,你可以用一...
    99+
    2023-05-16
    C++ 运算符重载 C++ 运算符 C++ 重载
  • c语言重载运算符怎么实现
    C语言不支持重载运算符,只能通过函数来实现类似的功能。例如,可以通过定义一个函数来代替重载的加法运算符: #include ty...
    99+
    2023-10-25
    c语言
  • C++运算符重载的详细讲解
    加号运算符重载 对于内置数据类型,编译器知道如何运算 但是对于自己封装的类,编译器无法进行运算 这时可以通过自己定义运算符重载进行运算 operator+ 通过成员函数重载+号 ...
    99+
    2022-11-12
  • C++双目运算符+=的重载详解
    目录1、+=重载2、friend重载+=3、运算符3.1 单目运算符3.2 双目运算符3.3 三目运算符4、重载++和重载- -总结1、+=重载 class Complex { ...
    99+
    2022-11-12
  • C语言运算符的重载实例分析
    本篇内容介绍了“C语言运算符的重载实例分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!写一个Add函数我们先讨论下面代码,并复习前面的内容...
    99+
    2023-06-26
  • Go语言实现运算符重载的方法详解
    目录前言使用实现总结前言 先带来日常的 GScript 更新:新增了可变参数的特性,语法如下: int add(string s, int ...num){ p...
    99+
    2022-11-11
  • C++中运算符重载问题详解
    C++中运算符重载问题详解运算符重载是C++中的一个重要特性,它允许我们为自定义的类类型重载运算符,使得这些类对象之间的运算能够像内置类型一样进行。在本文中,我们将详细讨论C++中运算符重载的概念、使用方法以及常见的问题。一、什么是运算符重...
    99+
    2023-10-22
    问题详解 C++运算符重载
  • C++超详细讲解运算符重载
    目录概念赋值运算符重载const成员取地址及const取地址操作符重载概念 C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类 型,函数名...
    99+
    2022-11-13
  • C++中的运算符重载问题详解
    C++中的运算符重载问题详解运算符重载是C++中一个强大而有用的特性,通过重载运算符,可以使得某个类的对象可以像基本类型数据一样使用各种运算符,方便地进行各种操作。本文将详细解释运算符重载的概念,并提供具体的代码示例。在C++中,运算符重载...
    99+
    2023-10-22
    运算符重载 (Operator Overloading) C++中的运算符 (Operators in C++) 重载运
  • Python的运算符重载详解
    目录一、前言二、重载作用三、自定义列表一、前言 运算符重载:为运算符定义方法 所谓重载,就是赋予新的含义同一个运算符可以有不同的功能 二、重载作用 让自定义的实例像内建对象一样进行运...
    99+
    2022-11-12
  • C++运算符重载详情介绍
    文章转自公众号:Coder梁(ID:Coder_LT) C++当中除了函数可以重载之外,其实运算符也是可以重载的。我们之前已经接触过一些,可能大家没有意识到。 举个例子,乘号*,运用...
    99+
    2022-11-12
  • C++基础知识之运算符重载详解
    目录运算符重载方式一, 使用成员函数重载运算符 需求:把牛肉换猪肉, 羊肉换猪肉方式二, 使用非成员函数【友元函数】重载运算符两种方式的区别两种方式的选择:总结运算符重载 为什么要使...
    99+
    2022-11-13
  • C++类与对象之运算符重载详解
    目录运算符重载加号运算符重载左移运算符重载递增运算符重载递减运算符重载赋值运算符重载关系运算符重载函数调用运算符重载总结运算符重载 运算符重载概念:对已有的运算符重新进行定义,赋予其...
    99+
    2022-11-12
  • 详解C++编程中一元运算符的重载
    可重载的一元运算符如下: !(逻辑“非”) &(取址) ~(二进制反码) *(取消指针引用) +(一元加) -(一元求反) ++(递增) --(递减) 转换运算符 后...
    99+
    2022-06-04
    详解 运算符
  • 详解C++ 运算符重载中返回值的坑
    相信不少朋友在学习运算符重载的时候,都会被参数与返回值应该是左值引用,还是右值引用,还是const常量所困扰。当然我无法一一枚举,这次先讲一下返回值的坑 (没错就是我亲手写的bug)...
    99+
    2022-11-12
  • 详解C++中递增运算符重载的实现
    目录递增运算符递增运算符重载的实现左移运算符重载前置递增运算符重载后置递增运算符重载递增运算符 形如"a++"、"++a"。如果a初始值为1,...
    99+
    2022-11-13
  • C++运算符重载怎么理解
    这期内容当中小编将会给大家带来有关C++运算符重载怎么理解,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。C++当中除了函数可以重载之外,其实运算符也是可以重载的。我们之前已经接触过一些,可能大家没有意识到...
    99+
    2023-06-22
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作