C++运算符重载——重载二元运算符
1、重载二元操作符的方法
二元运算符又称为双目运算符,即需要2个操作数的运算符,例如 + - * / 等。
运算符重载可以分为3种方式:类的非静态成员函数、类的友元函数、普通函数。
例如有 2 个操作数 a 和 b,二元运算符 ? (表示一个二元运算符),a ? b 的操作会被解释为下面2种形式之一
- //a ? b
- a.operator?(b); //类的非静态成员函数
- operator?(a, b); //友元函数 和 普通函数
第一种形式是运算符被重载为类的非静态成员函数,
这种方式要求运算符左边的的操作数(即第一个操作数a)必须是一个对象,operator?是这个对象的非静态成员函数
并且只能有一个参数。
第二种形式是运算符被重载为类的友元函数 或 普通函数,
这种方式需要2个参数,
重载为 类的友元函数 和 普通函数的区别是 类的友元函数可以直接访问类的私有成员,而普通函数不可以。
2、应用举例(对象 ? 对象)
下例中有3个complex类 ComplexA、ComplexB 和 ComplexC,3个类都重载了加减乘除 运算符。
其中ComplexA使用类的非静态成员函数方式重载,ComplexB使用类的友元函数方式重载,ComplexC使用普通函数方式重载。
需要注意的是复数的加减乘除运算的算法是有问题的,只是一个说明重载方法的例子,
另外重载函数的参数最好使用const关键字限定,至于返回值是否用const限定,需要取决于你的设计,比如允许C3 = ++(C1+C2)这种情况,就不能用cosnt限定。
至于不同类型的对象间的操作,通常是没有意义的。
- #include <iostream>
- using namespace std;
- class ComplexA
- {
- public:
- //默认构造函数(Default constructor)
- ComplexA(){cout<<"Default Constructor"<<endl;}
- //带参数的构造函数(The constructor with parameters)
- ComplexA(double re, double im):real(re),image(im){cout<<"Parameter Constructor"<<endl;}
- //拷贝构造函数(Copy constructor)
- ComplexA(const ComplexA& ref){real = ref.real; image = ref.image; cout<<"Copy Constructor"<<endl;}
- //析构函数(destructor)
- ~ComplexA(){cout<<"Destructor"<<endl;}
- //Operator Overload : +
- ComplexA operator+(ComplexA& ref)
- {
- return ComplexA(real + ref.real, image + ref.image);
- }
- //Operator Overload : -
- ComplexA operator-(ComplexA& ref)
- {
- return ComplexA(real - ref.real, image - ref.image);
- }
- //Operator Overload : *
- ComplexA operator*(ComplexA& ref)
- {
- return ComplexA(real * ref.real, image * ref.image);
- }
- //Operator Overload : /
- ComplexA operator/(ComplexA& ref)
- {
- return ComplexA(real / ref.real, image / ref.image);
- }
- //display
- void display(void){cout<<real<<"+"<<image<<"i"<<endl;}
- private:
- double real; //复数的实部
- double image; //复数的虚部
- };
- class ComplexB
- {
- public:
- //默认构造函数(Default constructor)
- ComplexB(){cout<<"Default Constructor"<<endl;}
- //带参数的构造函数(The constructor with parameters)
- ComplexB(double re, double im):real(re),image(im){cout<<"Parameter Constructor"<<endl;}
- //拷贝构造函数(Copy constructor)
- ComplexB(const ComplexB& ref){real = ref.real; image = ref.image; cout<<"Copy Constructor"<<endl;}
- //析构函数(destructor)
- ~ComplexB(){cout<<"Destructor"<<endl;}
- //Operator Overload : +
- friend ComplexB operator+(ComplexB& ref1, ComplexB& ref2)
- {
- return ComplexB(ref1.real + ref2.real, ref1.image + ref2.image);
- }
- //Operator Overload : -
- friend ComplexB operator-(ComplexB& ref1, ComplexB& ref2)
- {
- return ComplexB(ref1.real - ref2.real, ref1.image - ref2.image);
- }
- //Operator Overload : *
- friend ComplexB operator*(ComplexB& ref1, ComplexB& ref2)
- {
- return ComplexB(ref1.real * ref2.real, ref1.image * ref2.image);
- }
- //Operator Overload : /
- friend ComplexB operator/(ComplexB& ref1, ComplexB& ref2)
- {
- return ComplexB(ref1.real / ref2.real, ref1.image / ref2.image);
- }
- //display
- void display(void){cout<<real<<"+"<<image<<"i"<<endl;}
- private:
- double real; //复数的实部
- double image; //复数的虚部
- };
- class ComplexC
- {
- public:
- //默认构造函数(Default constructor)
- ComplexC(){cout<<"Default Constructor"<<endl;}
- //带参数的构造函数(The constructor with parameters)
- ComplexC(double re, double im):real(re),image(im){cout<<"Parameter Constructor"<<endl;}
- //拷贝构造函数(Copy constructor)
- ComplexC(const ComplexC& ref){real = ref.real; image = ref.image; cout<<"Copy Constructor"<<endl;}
- //析构函数(destructor)
- ~ComplexC(){cout<<"Destructor"<<endl;}
- //Get Data
- double GetReal(void){return real;}
- double GetImage(void){return image;}
- //display
- void display(void){cout<<real<<"+"<<image<<"i"<<endl;}
- private:
- double real; //复数的实部
- double image; //复数的虚部
- };
- //Operator Overload : +
- ComplexC operator+(ComplexC& ref1, ComplexC& ref2)
- {
- return ComplexC(ref1.GetReal() + ref2.GetReal(), ref1.GetImage() + ref2.GetImage());
- }
- //Operator Overload : -
- ComplexC operator-(ComplexC& ref1, ComplexC& ref2)
- {
- return ComplexC(ref1.GetReal() - ref2.GetReal(), ref1.GetImage() - ref2.GetImage());
- }
- //Operator Overload : *
- ComplexC operator*(ComplexC& ref1, ComplexC& ref2)
- {
- return ComplexC(ref1.GetReal() * ref2.GetReal(), ref1.GetImage() * ref2.GetImage());
- }
- //Operator Overload : /
- ComplexC operator/(ComplexC& ref1, ComplexC& ref2)
- {
- return ComplexC(ref1.GetReal() / ref2.GetReal(), ref1.GetImage() / ref2.GetImage());
- }
- int main(void)
- {
- ComplexA C1(,), C2(, ), C3;
- C3 = C1 + C2; C3.display();
- C3 = C1 - C2; C3.display();
- C3 = C1 * C2; C3.display();
- C3 = C1 / C2; C3.display();
- cout <<"--------------------------------------"<<endl;
- ComplexB C4(,), C5(, ), C6;
- C6 = C4 + C5; C6.display();
- C6 = C4 - C5; C6.display();
- C6 = C4 * C5; C6.display();
- C6 = C4 / C5; C6.display();
- cout <<"--------------------------------------"<<endl;
- ComplexC C7(,), C8(, ), C9;
- C9 = C7 + C8; C9.display();
- C9 = C7 - C8; C9.display();
- C9 = C7 * C8; C9.display();
- C9 = C7 / C8; C9.display();
- return ;
- }
3、应用举例(对象 ? 基本数据类型 or 基本数据类型 ? 对象)
上面的例子中是对象 和 对象之间的运算符重载,如果需要一个是对象 + char/int/float/double,或者反过来 char/int/float/double + 对象,这时上面的程序的重载方式就不适用了。
需要定义新的重载,如下列程序所示。
- #include <iostream>
- using namespace std;
- class ComplexD
- {
- public:
- ComplexD(double re = , double im = ):real(re),image(im){}
- ComplexD operator+(ComplexD& ref){return ComplexD(real+ref.real, image+ref.image);};
- ComplexD operator+(int a){cout<<"IN\t int \t\t";return ComplexD(real+a, image);};
- ComplexD operator+(double d){cout<<"IN\t double \t";return ComplexD(real+d, image);};
- ComplexD operator+(float f){cout<<"IN\t float \t\t";return ComplexD(real+f, image);};
- void display(void){cout<<real<<"+"<<image<<"i"<<endl;}
- double GetReal(void){return real;}
- double GetImage(void){return image;}
- private:
- double real;
- double image;
- };
- ComplexD operator+(int a, ComplexD& ref){cout<<"OUT\t int \t\t";return ComplexD(ref.GetReal()+a, ref.GetImage());};
- ComplexD operator+(double d, ComplexD& ref){cout<<"OUT\t double \t";return ComplexD(ref.GetReal()+d, ref.GetImage());};
- ComplexD operator+(float f, ComplexD& ref){cout<<"OUT\t float \t\t";return ComplexD(ref.GetReal()+f, ref.GetImage());};
- int main(void)
- {
- ComplexD D1(,), D2;
- D2 = D1 + ; D2.display();
- D2 = D1 + 2.1f; D2.display();
- D2 = D1 + 2.1; D2.display();
- D2 = +D1; D2.display();
- D2 = 2.1f + D1; D2.display();
- D2 = 2.1 +D1; D2.display();
- return ;
- }
C++运算符重载——重载二元运算符的更多相关文章
- C++ 重载运算符和重载函数
C++ 重载运算符和重载函数 C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载. 重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是 ...
- JavaScript一元运算符、二元运算符和三元运算符
在JavaScript中,运算符可以根据其实际操作数的个数进行分类. JavaScript中的大多数运算符是一个二元运算符(binary operator),将两个表达式合并成为一个稍复杂的表达式.譬 ...
- C++ 运算符重载一(二元运算符重载)
//二元运算符重载 #include<iostream> using namespace std; class Point { public: Point(int x,int y){ th ...
- C++之运算符重载(二元)
一.加号+ 1.成员函数重载 2.友元函数重载 二.输出符号<< 三.索引符号 [ ] 四.补充说明 1.<二元运算符重载>课程评论: (一)为什么<<运算符的重载 ...
- [置顶] C++基础之六:运算符的重载
网上太多有关运算符的重载了,但是写的太过的详细,不适合新手入门,特别是那什么++和--的前增量后增量重载,一元二元运算符重载,特殊运算符,下标运算符,new和delete,甚至是指针运算符的重载,吓退 ...
- C++解析七-重载运算符和重载函数
重载运算符和重载函数C++ 允许在同一作用域中的某个函数和运算符指定多个定义,分别称为函数重载和运算符重载.重载声明是指一个与之前已经在该作用域内声明过的函数或方法具有相同名称的声明,但是它们的参数列 ...
- C++ 自增、自减运算符的重载和性能分析
01 ++.--运算符重载函数的格式 自增运算符和自减运算符是有前置和后置之分的,如: a++ // 后置自增运算符 ++a // 前置自增运算符 b-- // 后置自减运算符 --b // 前置自减 ...
- C++基础之自增和自减运算符的重载
1. 格式 1.1 分为前置和后置格式: int x = 0; int y = 0; // 后置自增运算符 x++; // 前置自增运算符 ++x; // 后置自减运算符 y--; // 前置自减运算 ...
- C++ Primer : : 第十四章 : 重载运算符与类型转换之类型转换运算符和重载匹配
类型转换运算符 class SmallInt { public: SmallInt(int i = 0) : val(i) { if (i < 0 || i > 255) throw st ...
随机推荐
- django概述
一.django的特点 1.提供一体化的web解决方案,什么叫一体化:mvc 2.等你玩儿牛逼了可以拔插组件,换成自己顺手或者更牛逼的组件
- trap命令使用
分享一个shell脚本技巧,大家写shell脚本的时候,一般而言仅仅保证功能可用,但程序的鲁棒性却不是太好,不够健壮,多数是脚本处理 一些中断信号导致,应对非预期的系统信号,其实系统自带的trap命令 ...
- 关于使用regsvr32命令注册ActiveX控件失败的解决办法
昨天小编也遇到这样问题,步骤一切都对,没有错误,但是每次在命令行下输入的时候,都会弹出一个对话框: 最后我发现是存放ActiveX控件的路径中带有中文文件名字所导致,所以导致的错误,我们将所在路径下的 ...
- Factorial Trailing Zeroes
Given an integer n, return the number of trailing zeroes in n!. Note: Your solution should be in log ...
- java笔试题(2)
简述构造器的运行机制 首先要注意的是的构造器并不是函数,所以他并不能被继承,这在我们extends的时候写子类的构造器时比较的常见,即使子类构造器参数和父类的完全一样,我们也要写super就是因为这个 ...
- kibana 修改Ico图标
修改此路径下的E:\happy\kinbana\kibana-4.2.2-windows\kibana-4.2.2-windows\optimize\bundles的commons.bundle.js ...
- 【Merge Sorted Array】cpp
题目: Given two sorted integer arrays nums1 and nums2, merge nums2 into nums1 as one sorted array. Not ...
- CSS3动画制作的简单示例
CSS3 大大强化了制作动画的能力,但是如果要做出图案比较复杂的动画,选择 GIF 依然是一个不错的选择.今天给大家介绍一个使用 CSS animation 配合雪碧图(CSS sprite)来制作动 ...
- Bootstrap的宽度和分辨率的差别
首先在bootstrap里面所有的样式并在pc上是根据px的单位来判断的,就是我们说的分辨率, @media(min-width:1200px){ ......里面的样式 } 那么就是说当你的屏幕放大 ...
- NYOJ-85 有趣的数 AC 分类: NYOJ 2014-01-17 21:42 240人阅读 评论(0) 收藏
这道题目就是,找规律,小学奥数,找规律不难吧, #include<stdio.h> int sc(int x); int main(){ int n=0; int num,cs,k; sc ...