【C++之运算符重载1】复数类 Complex 重载运算符 “+”

简介: 【C++之运算符重载1】复数类 Complex 重载运算符 “+”

题目要求


定义一个复数类 Complex ,重载运算符 “+” ,使之能用于复数的加法运算。参加运算的两个运算量可以都是复数类对象,也可以其中一个是整数。例如:c1+c2, c1+i, i+c1 均合法(设 i 为整数,c1、c2为复数)。编程序,分别求两个复数之和、整数和复数之和。


——谭浩强的《C++面向对象程序设计》第4章习题第3小题


多态性


多态性是面向对象程序设计的一个重要特性。


多态性是指不同的对象在接收同样消息时可以表现出不同的行为特征。

消息在C++程序中是指对函数的调用信息。因此多态的本质是指同样的函数在不同对象调用时会产生不同的功能表现。


多态性可分为静态多态性和动态多态性两大类。


函数重载和运算符重载属于静态多态性,在编译程序时系统就可确定具体调用哪个函数,因此静态多态性又称编译时的多态性。静态多态性是通过函数重载实现的。

动态多态性是在程序运行时才能确定函数操作所针对的对象。它又称运行时的多态性。动态多态性是通过虚函数实现的。


运算符重载


对已有的运算符赋予新的含义,用一个运算符表示不同功能的运算,这就是运算符重载。


运算符重载实质上是函数的重载。编译系统对重载运算符的选择,遵循函数重载的选择原则。


运算符重载函数的定义格式是:


重载函数值的数据类型 operator 运算符 (形参表)
{ 重载处理 }


函数名是由 operator 和 运算符 组成的。


规则和限制

C++中可以重载除下列运算符外的所有运算符:


.(成员访问)   
  ::(域)   
  *(成员指针访问)   
  ?:(条件)   
  sizeof(长度)


只能重载C++语言中已有的运算符,不可臆造新的。

不能改变原运算符的优先级、结合性和功能类型/功能范围。

不能改变原有的操作数个数;运算符重载不能带默认参数。

经重载的运算符,其操作数中至少应该有一个是自定义类型。

运算符重载有两种形式:成员函数与友元函数。

如果函数需要访问类的私有成员,则必须声明为友元函数。

一般来说,重载单目运算符,通常重载为类的成员函数;重载双目运算符,通常重载为类的友元函数。


重载为类的成员函数:


格式:


<类名> operator <运算符> (<参数表>)
{…}


如果将运算符重载函数作为成员函数,它可以通过 this 指针访问本类的数据成员,因此可以少写一个函数的参数。


单目运算符采用成员函数形式重载时, 该<参数表>无参数;


双目运算符采用成员函数形式重载时,该<参数表>中有一个参数。

例如:


Complex operator + (Complex &c)
// 形参是 Complex 类对象的引用,要求实参为 Complex 类对象
{
  return Complex(real + c.real, imag + c.imag);
  // 注意在表达式中重载的运算符 "+" 右侧应为 Complex 类的对象
  // 比如 c3 = c1 + c2
}


“ + ” 是双目运算符,本来需要两个参数,但有一个参数是隐含的,运算符函数是用 this 指针隐式访问类对象的成员。所以其实重载函数 operator + 访问了两个对象中的成员,一个是 this 指针指向的对象中的成员,一个是形参对象中的成员。


重载为类的友元函数:


格式:


friend <类型> operator <运算符> (<参数表>) 
{…}


以下运算符不能重载为友元函数,必须重载为成员函数:


=  ()  [ ]    ->


单目运算符被重载为友元函数时, 该<参数表>有一个参数;

双目运算符被重载为友元函数时,该<参数表>有两个参数,形参的顺序任意,不要求第一个参数必须为类对象。但在使用运算符的表达式中,要求运算符左侧的操作数与函数第一个参数对应,运算符右侧的操作数与函数的第二个参数对应。

例如:


friend Complex operator + (int &, Complex &)
{
  return Complex(i + c.real, c.imag);
  // 注意在表达式中重载的运算符 "+" 左侧应为 int 类对象,右侧应为 Complex 类的对象
  // 比如 c2 = i + c1
}


程序


/*
*************************************************************************
@file:    main.cpp
@date:   2020.12.3
@author: Xiaoxiao
@brief:   复数类 Complex 重载运算符 “+”
@blog:    https://blog.csdn.net/weixin_43470383/article/details/110496946
*************************************************************************
*/
#include <iostream>
using namespace std;
class Complex 
{
public:
  // 定义一个无参的构造函数
  Complex(){real = 0; imag = 0;}
  // 定义构造函数,并用参数初始化表对其数据成员初始化
  Complex(double r, double i):real(r),imag(i){}
  void display();
  // 声明重载运算符 "+" 的函数
  // 重载函数为成员函数
  Complex operator + (Complex &c); // c1 + c2
  // 形参是Complex类对象的引用,要求实参为Complex类对象
  Complex operator + (int &i); // c1 + i
  // 重载函数为友元函数
  friend Complex operator + (int &, Complex &); // i + c
private:
  double real; // 复数实部
  double imag; // 复数虚部
};
Complex Complex::operator + (Complex &c) // 注意类外定义成员函数需要限定作用域
{
  cout << "Complex operator + (Complex &c)" << endl;
  return Complex(real + c.real, imag + c.imag);
  // 注意在表达式中重载的运算符 "+" 右侧应为 Complex 类的对象
  // 比如 c3 = c1 + c2
}
Complex Complex::operator + (int &i)
{
  cout << "Complex operator + (int &i)" << endl;
  return Complex(real + i, imag);
  // 注意在表达式中重载的运算符 "+" 左侧应为 Complex 类的对象
  // 比如 c2 = c1 + i
}
Complex operator + (int &i, Complex &c)
{
  cout << "Complex operator + (int &i, Complex &c)" << endl;
  return Complex(i + c.real, c.imag);
  // 注意在表达式中重载的运算符 "+" 左侧应为 int 类对象,右侧应为 Complex 类的对象
  // 比如 c2 = i + c1
}
void Complex::display()
{
  cout << "(" << real << ", " << imag << ")" << endl;
}
int main()
{
  Complex c1(5, -2), c2(-4, 3), c3;
  int i = 6;
  c3 = c1 + i;
  cout << "c1 + i = " << endl;
  c3.display();
  c3 = i + c1;
  cout << "i + c1 = " << endl;
  c3.display();
  c3 = c1 + c2;
  cout << " c1 + c2 = " << endl;
  c3.display();
  system("pause");
  return 0;
}


运行结果



输出:

Complex operator + (int &i)

c1 + i =

(11, -2)

Complex operator + (int &i, Complex &c)

i + c1 =

(11, -2)

Complex operator + (Complex &c)

c1 + c2 =

(1, 1)


从提示信息可以看出表达式重载运算符时调用了哪个函数。


相关文章
|
11月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
268 5
|
C++
【C++基础】运算符详解
这篇文章详细解释了C++中运算符的用法,包括算术运算符、赋值运算符、比较运算符和逻辑运算符,以及它们在表达式中的作用和示例。
173 2
C++(十九)new/delete 重载
本文介绍了C++中`operator new/delete`重载的使用方法,并通过示例代码展示了如何自定义内存分配与释放的行为。重载`new`和`delete`可以实现内存的精细控制,而`new[]`和`delete[]`则用于处理数组的内存管理。不当使用可能导致内存泄漏或错误释放。
C++(十五) 运算符重载
C++中的运算符重载允许对已有运算符的功能进行重新定义,从而扩展语言功能、简化代码并提升效率。重载遵循特定语法,如 `friend 类名 operator 运算符(参数)`。重载时需注意不可新增或改变运算符数量、语义、优先级、结合性和返回类型。常见示例包括双目运算符 `+=` 和单目运算符 `-` 及 `++`。输入输出流运算符 `&lt;&lt;` 和 `&gt;&gt;` 也可重载。部分运算符只能作为成员函数重载。
|
C++
c++学习笔记02 运算符
C++学习笔记,介绍了C++中的运算符,包括基本的加减乘除、求模、前后置递增递减、赋值运算符、比较运算符和逻辑运算符的使用及其注意事项。
110 6
|
NoSQL 编译器 Redis
c++开发redis module问题之如果Redis加载了多个C++编写的模块,并且它们都重载了operator new,会有什么影响
c++开发redis module问题之如果Redis加载了多个C++编写的模块,并且它们都重载了operator new,会有什么影响
|
存储 C++
【C++】string类的使用③(非成员函数重载Non-member function overloads)
这篇文章探讨了C++中`std::string`的`replace`和`swap`函数以及非成员函数重载。`replace`提供了多种方式替换字符串中的部分内容,包括使用字符串、子串、字符、字符数组和填充字符。`swap`函数用于交换两个`string`对象的内容,成员函数版本效率更高。非成员函数重载包括`operator+`实现字符串连接,关系运算符(如`==`, `&lt;`等)用于比较字符串,以及`swap`非成员函数。此外,还介绍了`getline`函数,用于按指定分隔符从输入流中读取字符串。文章强调了非成员函数在特定情况下的作用,并给出了多个示例代码。
|
NoSQL Redis C++
c++开发redis module问题之避免多个C++模块之间因重载operator new而产生的冲突,如何解决
c++开发redis module问题之避免多个C++模块之间因重载operator new而产生的冲突,如何解决
|
自然语言处理 程序员 C++
C++基础知识(五:运算符重载)
运算符重载是C++中的一项强大特性,它允许程序员为自定义类型(如类或结构体)重新定义标准运算符的行为,使得这些运算符能够适用于自定义类型的操作。这样做可以增强代码的可读性和表达力,使得代码更接近自然语言,同时保持了面向对象编程的封装性。
137 0
|
8月前
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。

热门文章

最新文章