【C++深度剖析学习总结】 23 操作符重载

举报
CodeAllen 发表于 2021/10/29 22:59:04 2021/10/29
【摘要】 所谓重载,就是赋予新的含义。函数重载(Function Overloading)可以让一个函数名有多种功能,在不同情况下进行不同的操作。运算符重载(Operator Overloading)也是一个道理,同一个运算符可以有不同的功能。 实际上,我们已经在不知不觉中使用了运算符重载。例如,+号可以对不同类型(int、float 等)的数据...

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

实际上,我们已经在不知不觉中使用了运算符重载。例如,+号可以对不同类型(int、float 等)的数据进行加法操作;<<既是位移运算符,又可以配合 cout 向控制台输出数据。C++ 本身已经对这些运算符进行了重载。

C++ 也允许程序员自己重载运算符,这给我们带来了很大的便利。

 

下面的代码定义了一个复数类,通过运算符重载,可以用+号实现复数的加法运算:


  
  1. #include <iostream>
  2. using namespace std;
  3. class complex{
  4. public:
  5. complex();
  6. complex(double real, double imag);
  7. public:
  8. //声明运算符重载
  9. complex operator+(const complex &A) const;
  10. void display() const;
  11. private:
  12. double m_real; //实部
  13. double m_imag; //虚部
  14. };
  15. complex::complex(): m_real(0.0), m_imag(0.0){ }
  16. complex::complex(double real, double imag): m_real(real), m_imag(imag){ }
  17. //实现运算符重载
  18. complex complex::operator+(const complex &A) const{
  19. complex B;
  20. B.m_real = this->m_real + A.m_real;
  21. B.m_imag = this->m_imag + A.m_imag;
  22. return B;
  23. }
  24. void complex::display() const{
  25. cout<<m_real<<" + "<<m_imag<<"i"<<endl;
  26. }
  27. int main(){
  28. complex c1(4.3, 5.8);
  29. complex c2(2.4, 3.7);
  30. complex c3;
  31. c3 = c1 + c2;
  32. c3.display();
  33. return 0;
  34. }

运行结果:
6.7 + 9.5i

本例中义了一个复数类 complex,m_real 表示实部,m_imag 表示虚部,第 10 行声明了运算符重载,第 21 行进行了实现(定义)。认真观察这两行代码,可以发现运算符重载的形式与函数非常类似。

运算符重载其实就是定义一个函数,在函数体内实现想要的功能,当用到该运算符时,编译器会自动调用这个函数。也就是说,运算符重载是通过函数实现的,它本质上是函数重载。

运算符重载函数除了函数名有特定的格式,其它地方和普通函数并没有区别。

 

上面的运算符重载还可以有更加简练的定义形式:


  
  1. complex complex::operator+(const complex &A)const{
  2. return complex(this->m_real + A.m_real, this->m_imag + A.m_imag);
return 语句中的

 
complex(this->m_real + A.m_real, this->m_imag + A.m_imag)
 
会创建一个临时对象,这个对象没有名称,是一个匿名对象。在创建临时对象过程中调用构造函数,return 语句将该临时对象作为函数返回值。
 

在全局范围内重载运算符

运算符重载函数不仅可以作为类的成员函数,还可以作为全局函数。更改上面的代码,在全局范围内重载+,实现复数的加法运算:


  
  1. #include <iostream>
  2. using namespace std;
  3. class complex{
  4. public:
  5. complex();
  6. complex(double real, double imag);
  7. public:
  8. void display() const;
  9. //声明为友元函数
  10. friend complex operator+(const complex &A, const complex &B);
  11. private:
  12. double m_real;
  13. double m_imag;
  14. };
  15. complex operator+(const complex &A, const complex &B);
  16. complex::complex(): m_real(0.0), m_imag(0.0){ }
  17. complex::complex(double real, double imag): m_real(real), m_imag(imag){ }
  18. void complex::display() const{
  19. cout<<m_real<<" + "<<m_imag<<"i"<<endl;
  20. }
  21. //在全局范围内重载+
  22. complex operator+(const complex &A, const complex &B){
  23. complex C;
  24. C.m_real = A.m_real + B.m_real;
  25. C.m_imag = A.m_imag + B.m_imag;
  26. return C;
  27. }
  28. int main(){
  29. complex c1(4.3, 5.8);
  30. complex c2(2.4, 3.7);
  31. complex c3;
  32. c3 = c1 + c2;
  33. c3.display();
  34. return 0;
  35. }

 

1.需要解决的问题

下面的复数解决方案是否可行?

复数的加法操作


  
  1. #include <stdio.h>
  2. class Complex
  3. {
  4. int a;//hide variable
  5. int b;
  6. public:
  7. Complex(int a = 0, int b = 0)
  8. {
  9. this->a = a;
  10. this->b = b;
  11. }
  12. int getA()
  13. {
  14. return a;
  15. }
  16. int getB()
  17. {
  18. return b;
  19. }
  20. friend Complex Add(const Complex& p1, const Complex& p2);
  21. };
  22. Complex Add(const Complex& p1, const Complex& p2)
  23. {
  24. Complex ret;
  25. ret.a = p1.a + p2.a;
  26. ret.b = p1.b + p2.b;
  27. return ret;
  28. }
  29. int main()
  30. {
  31. Complex c1(1, 2);
  32. Complex c2(3, 4);
  33. Complex c3 = Add(c1, c2); // c1 + c2
  34. printf("c3.a = %d, c3.b = %d\n", c3.getA(), c3.getB());
  35. return 0;
  36. }

 

2.思考

Add函数可以解决Complex对象相加的问题,但是Complex是现实世界中确实存在的复数,并且复数在数学中的地位和普通的实数相同。

为什么不让+操作符也支持复数相加呢?

 

3.操作符重载

C++中的重载能够扩展操作符的功能

操作符的重载以函数的方式进行

 

本质:

用特殊形式的函数扩展操作符的功能

通过operator关键字可以定义特殊的函数

operator的本质是通过函数重载操作符

 

语法:

Sign为系统中预定义的操作符,如:+,-,*,/,等

 

操作符重载初探--全局函数版本的实现


  
  1. #include <stdio.h>
  2. class Complex
  3. {
  4. int a;
  5. int b;
  6. public:
  7. Complex(int a = 0, int b = 0)
  8. {
  9. this->a = a;
  10. this->b = b;
  11. }
  12. int getA()
  13. {
  14. return a;
  15. }
  16. int getB()
  17. {
  18. return b;
  19. }
  20. friend Complex operator + (const Complex& p1, const Complex& p2);//友元函数
  21. };//类定义之后又;
  22. Complex operator + (const Complex& p1, const Complex& p2)
  23. {
  24. Complex ret;
  25. ret.a = p1.a + p2.a;
  26. ret.b = p1.b + p2.b;
  27. return ret;
  28. }
  29. int main()
  30. {
  31. Complex c1(1, 2);
  32. Complex c2(3, 4);
  33. Complex c3 = c1 + c2; // operator + (c1, c2)
  34. printf("c3.a = %d, c3.b = %d\n", c3.getA(), c3.getB());
  35. return 0;
  36. }

 

可以将操作符重载函数定义为类的成员函数

  • - 比全局操作符重载函数少一个参数(左操作数)this充当左操作数

  • - 不需要依赖友元就可以完成操作符重载

  • - 编译器优先在成员函数中寻找操作符重载函数

成员函数重载操作符--成员函数调用


  
  1. #include <stdio.h>
  2. class Complex
  3. {
  4. int a;
  5. int b;
  6. public:
  7. Complex(int a = 0, int b = 0)
  8. {
  9. this->a = a;
  10. this->b = b;
  11. }
  12. int getA()
  13. {
  14. return a;
  15. }
  16. int getB()
  17. {
  18. return b;
  19. }
  20. Complex operator + (const Complex& p) //成员函数版本---优先调用
  21. {
  22. Complex ret;
  23. printf("Complex operator + (const Complex& p)\n");
  24. ret.a = this->a + p.a;
  25. ret.b = this->b + p.b;
  26. return ret;
  27. }
  28. friend Complex operator + (const Complex& p1, const Complex& p2);
  29. };
  30. Complex operator + (const Complex& p1, const Complex& p2) //全局函数版本
  31. {
  32. Complex ret;
  33. printf("Complex operator + (const Complex& p1, const Complex& p2)\n");
  34. ret.a = p1.a + p2.a;
  35. ret.b = p1.b + p2.b;
  36. return ret;
  37. }
  38. int main()
  39. {
  40. Complex c1(1, 2);
  41. Complex c2(3, 4);
  42. Complex c3 = c1 + c2; // c1.operator + (c2) ,通过成员函数实现
  43. printf("c3.a = %d, c3.b = %d\n", c3.getA(), c3.getB());
  44. return 0;
  45. }

 

 

小结

操作符重载是C++的强大特性之一

操作符重载的本质是通过函数扩展操作符的功能

operator关键字是实现操作符重载的关键

操作符重载遵循相同的函数重载规则

全局函数和成员函数都可以实现对操作符的重载

文章来源: allen5g.blog.csdn.net,作者:CodeAllen的博客,版权归原作者所有,如需转载,请联系作者。

原文链接:allen5g.blog.csdn.net/article/details/117452621

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。