第13章 运算符重载
13.1 运算符重载
说明:程序中出现的表达式 c1+c2 编译程序将其解释为: c1.operator+(c2); 其中,c1和c2是complex类的对象,而operator+( )是运算符 “+”的重载函数。 同理 c1-c2 解释为: c1.operator-(c2); 因此:A行 c4+=c1+c2; 被解释为: c4.operaotr+=(c1.operaotr+(c2));
说明: ★程序中的A行调用构造函数产生一个临时对象,并返回该 临时对象的值。 我们已经知道,对于双目运算符,重载为类的成员函数时, 仅有一个参数;重载为友元函数时,有两个参数,因此,当 程序中出现: c1+c2 编译系统将其解释为 operator+(c1, c2); 这将调用如下的函数进行求值运算: complex operator+(complex &c1, complex &c2);
(4)运算符的重载实际上是函数的重载。不同之处在于系 统约定了重载运算符的函数名,operator @( ); (5)不可臆造新的运算符。必须把重载运算符限制在C++ 语言中已有的运算符范围内的允许重载的运算符之中。
2. 重载为友元函数
运算符重载函数还可以为友元函数。当重载为友元函数 时,由于友元函数不是类的成员函数,尽管它可以访问类的 所有成员(包括私有成员),但必须通过类的对象才能实现 这种访问,因此当运算符重载为友元函数时,对于双目运算, 友元函数必须有2个参数,对于单目运算符,友元函数也必须 有一个参数。 注意,有些运算符不能重载为友元函数,它们是: =,(),[ ]和->。
Complex operator +(Complex &c1, Complex &c2) { Complex t; t.Real=c1.Real+c2.Real; t.Image=c1.Image+c2.Image; return t; } Complex operator+(Complex c1, float s) { Complex t; t.Real=c1.Real+s; t.Image=c1.Image; return t; }
C++基础系列——运算符重载
C++基础系列——运算符重载1. 运算符重载简介所谓重载,就是赋予新的含义。
函数重载(Function Overloading)可以让⼀个函数名有多种功能,在不同情况下进⾏不同的操作。
同样运算符重载(Operator Overloading)可以让同⼀个运算符可以有不同的功能。
可以对 int、float、string 等不同类型数据进⾏操作<< 既是位移运算符,⼜可以配合 cout 向控制台输出数据也可以⾃定义运算符重载:class Complex{public:Complex();Complex(double real, double imag);Complex operator+(const Complex &a) const;void display() const;private:double m_real;double m_imag;};// ...// 实现运算符重载Complex Complex::operator+(const Complex &A) const{Complex B;B.m_real = this->m_real + A.m_real;B.m_imag = this -> m_imag + A.m_imag;return B;// return Complex(this->m_real + A.m_real, this->m_imag + A.m_imag);}int main(){Complex c1(4.3, 5.8);Complex c2(2.7, 3.7);Complex c3;c3 = c1 + c2; // 运算符重载c3.display();return 0;}运算结果7 + 9.5i运算符重载其实就是定义⼀个函数,在函数体内实现想要的功能,当⽤到该运算符时,编译器会⾃动调⽤这个函数,它本质上是函数重载。
第12-13讲 运算符重载
7.1.2 运算符重载规则
可以重载C++中除下列运算符外的所有运算符: 类属关系运算符. 成员指针运算符* 作用域分辨符:: sizeof运算符 三目运算符?: 只能重载C++语言中已有的运算符,不可臆造新的。 不能改变操作数个数。 不能改变原运算符的优先级和结合性。 不能改变运算符对预定义类型数据的操作方式。 经重载的运算符,其操作数中至少应该有一个是自定 义类型。
14
void complex:: print()//显示输出复数 { cout<<real; if(imag>0) cout<<"+"; if(imag!=0) cout<<imag<<"i";cout<<endl; } void main() { complex A1(2.3,4.6),A2(3.6,2.8),A3,A4,A5,A6;//定义6个复数类对象 A1.print(); A2.print(); A3=A1+A2; A3.print(); A4=A1-A2; A4.print(); A5=A1*A2; A5.print(); A6=A1/A2; A6.print(); }
2
class complex { public: double real,imag; //定义一个实部,一个虚部变量 complex(double r=0,double i=0) { real=r; imag=I; } }; main() { complex com1(1.1,2.2),com2(3.3,4.4),total; total=com1+com2; 错误,2个自定义的数据类 return 0; 型complex类型对象相加, } C++无法实现. 希望实现total=com1+com2;重载“+”
作业13
HZAU-专业C++作业13(运算符重载)判断题1. 所有的运算符都可以重载。
F2.在C++中,编译时的多态性是通过函数和运算符的重载实现的。
T3. 运算符重载是通过对已有的运算符重新定义操作功能来实现的,它是C++语言多态性的一种表现。
T4. 运算符重载只可以是类的成员函数和友元函数,不可以是普通函数。
T5. 运算符重载是通过函数来实现的,定义运算符重载函数时,函数名中要使用关键字operator。
T6. 运算符重载后,优先级、结合性和操作数都是不变的。
T7. 重载运算符的函数也可以设置默认参数。
F8. 使用成员函数方法和使用友元函数的方法所定义的重载运算符函数的参数个数是不相同的。
T9. 运算符“.”、“::” 、“?:” 不能重载T10. 运算符=、()、[]和->可作为类成员运算符, 不能作为友员运算符。
T选择题1.下列运算符中,(A )运算符在C++中不能重载。
A. ?:B. [ ]C. newD. &&2.下列运算符不能用友元函数重载的是(c)。
A. +B. =C. *D. <<3.在一个类中可以对一个操作符进行( D )重载。
A. 1种B. 2种及以下C. 3种及以下D. 多种4.友元运算符obj1>obj2被C++编译器解释为( A )。
A. operator > (obj1, obj2)B. > (obj1, obj2)C. obj2.operator > (obj1)D. obj1.operator > (obj2)5.下列关于C++运算符函数的返回类型的描述中,错误的是( C )。
A. 可以是类类型B. 可以是int类型C. 可以是void类型D. 可以是float类型6.下列关于运算符重载的描述中,正确的是( D )。
A. 运算符重载可以改变运算符的操作数的个数B. 运算符重载可以改变优先级C. 运算符重载可以改变结合性D. 运算符重载不可以改变语法结构7.先加一然后再使用的++运算符的重栽形式是(A)。
C++电子教案第十三章 运算符重载
1-7
• • • • • •
联系方式:
运算符重载为类的友元函数
• 将重载的运算符函数定义为类的友元函数,称为友元运算符 函数。友元运算符函数不是类的成员,其在类内声明原型, 在类外定义函数本身。由于友元运算符函数不是类的成员函 数,不属于任何一个类对象,所以没有this指针,因此,重 载双目运算符时要有两个参数,重载单目运算符时只要一个 参数就可以了。一般来说,友元运算符函数在类内声明的一 般形式为: friend<返回类型> operator<运算符>(参数表); 在类外定义的一般形式为: <返回类型> operator<运算符>(参数表) { 函数体 }
运算符重载简介
•
简单来说,运算符重载是实现编译时的多态性的另外一 种形式。运算符重载是对已有的运算符赋予多重含义, 使同一个运算符作用于不同类型的数据时,导致不同类 型的行为。
运算符重载的实质是函数重载。事实上,C++语言中的 每一个运算符对应着一个运算符函数,在实现过程中, 把指定的运算表达式中的运算符转化为对运算符函数的 调用,而表达式中的运算对象转化为运算符函数的实参, 这个过程是在编译阶段完成的。
C++中为什么要支持运算符重载,其好处如何体现? 运算符重载的规则有哪些? 运算符重载有两种形式,其分别是什么? 写一个程序测试输入的长度能否构成一个三角形。构成 三角形的条件是任意两边之和大于第三边。
北京源智天下科技有限公司
1-12
联系方式:
•
北京源智天下科技有限公司
1-3
联系方式:
运算符重载的特点
•
尽管运算符重载是一种特殊的函数重载,但相比函数重 载,运算符重载有着自身的一些特点。使用operator关 键字对重载函数进行标识和定义。运算符有 3种形式, 即中缀、后缀、前缀,它们的 operator 表示形式如表 13-1所示。
运算符重载
1、多态性的基本概念2、派生类对象替换基类对象3、虚函数的定义4、抽象类的定义5、宠物类的设计6、运算符重载7、日期类对象判断大小8、分数类对象运算符重载☐运算符重载指赋予运算符新的操作功能,主要用于对类的对象的操作☐运算符+意味着多少对象类型的加法呢?☐还可以定义新的对象类型加法☐运算符重载定义形式:<类型><类名>::operator<操作符>(<参数表>){函数体}☐首先定义虚数类☐虚数可以描述为:a+bi☐a与b看成实数,定义成double类型☐成员函数除了构造与析构外,还有:☐输出虚数、修改虚数、得到实部a、得到虚部b ☐相加+、判相等==#include <iostream>using namespace std;class Complex{private:double real, imag;public:Complex(double r = 0, double i = 0): real(r), imag(i){ }double Real(){return real;}double Imag(){return imag;}Complex operator +(Complex&);Complex operator +(double);bool operator ==(Complex);~Complex(){ };Complex Complex::operator +(Complex &c)// 重载运算符+,两边是虚数对象{Complex temp;temp.real = real+c.real;temp.imag = imag+c.imag;return temp;}Complex Complex::operator +(double d)// 重载运算符+,左边是虚数对象,右边是双精度数{Complex temp;temp.real = real+d;temp.imag=imag;return temp;}bool Complex::operator ==(Complex c)// 重载运算符=={if (real == c.real && imag == c.imag)return true;elseint main(){Complex c1(3,4),c2(5,6),c3;cout << "C1 = " << c1.Real() << "+j" << c1.Imag() << endl;cout << "C2 = " << c2.Real() << "+j" << c2.Imag() << endl;c3 = c1+c2;cout << "C3 = " << c3.Real() << "+j" << c3.Imag() << endl;c3 = c3+6.5;cout << "C3 + 6.5 = " << c3.Real() << "+j" << c3.Imag() << endl;if ( c1==c2 )cout<<“两个复数相等”;elsecout<<“两个复数不相等”;return 0;☐运算符++分前置运算符和后置运算符☐例如: ++Y与Y++☐前置运算符定义Complex Complex::operator ++ () {real+=1;return *this;}☐后置运算符定义Complex Complex::operator ++ (int) {real+=1;return *this;}。
运算符重载的方法
运算符重载的方法
运算符重载是指在类中重新定义某个运算符,使其能够用于类的实例。
重载运算符可以方便地对类对象进行加、减、乘、除等基本运算,也可以定义自定义运算符,以实现类特定的功能。
运算符重载的方法如下:
1.编写运算符重载函数:将要重载的运算符的符号作为函数名,如“+”、“-”、“*”、“/”等,对于自定义运算符可以自己命名。
2.指定参数个数和类型:重载函数需要参数,这些参数有不同的类型,可以是类的成员变量或其他类型的变量。
3.重载运算符要求:
(1)操作数至少有一个是类的实例,重载函数可以定义为类的成员函数或全局函数。
(2)运算符重载不能改变原有操作数的类型或个数。
(3)运算符重载函数可以返回任何类型的值,甚至可以是类的实例。
4.使用运算符:一旦重载运算符函数定义好之后,就可以像使用普通的运算符一样使用它。
需要注意的是,运算符重载不是万能的,对于某些运算符,如“&&”、“”等,是不能够重载的。
此外,在使用运算符重载时应该避免过于复杂的实现方式,以防止代码难以理解和维护。
C 程序设计 教学课件 ppt 第13章_运算符重载
13.2 运算符重载为成员函数
Complex Complex::operator - () { return Complex(-real, -image);
} Complex Complex::operator + (const Complex &c) {
return Complex(real + c.real, image + c.image); } int main() {
• 运算符重载为类的成员函数时,函数的参数个数 比运算符原来的操作数少一个(后置++、--除外) ,这是因为重载为类的成员函数的话,该类的对象 使用此运算符时,相当于在操作数中隐式包含了指 向对象本身的this指针,自身的数据可以直接访问 ,这样就不需要再在参数表中写一个“this”进行 传递了
▫ 重载一元运算符时,操作数由指向对象的this指针提 供,因此不需要在参数表中提供任何参数;重载二元 运算符时,参数表中的唯一参数默认为右操作数,而 将this视为左操作数
)以及流输入输出操作符(<<, >>)都可以重载
C++程序设计,郑莉,清华大学 5
13.1 运算符重载的规则
• 重载运算符必须至少有一个类类型的操作数。而 用于内置类型的运算符的定义不可以改变。不能 为任何内置类型定义新的运算符,例如不能定义 两个数组的加法运算符
▫ 例如不可以重载用于整型的加号运算符: int operator + (int, int); //以上操作违法,不可 以重载用于内置类型的运算符
C++程序设计,郑莉,清华大学 19
13.3 运算符重载为非成员函数
运算符重载
24
运算符重载
思考:如果希望表达式i+c1有意义, 思考:如果希望表达式i+c1有意义,应该 i+c1有意义 如何定义重载运算符函数? 如何定义重载运算符函数?
friend Complex operator+(int &i,Complex &c); Complex operator+(int &i, Complex &c) {return Complex(i+c.real,c.imag);}
2
什么是运算符重载
代码分析: 代码分析: 例10.1 通过函数来实现复数 相加。 相加。
CCS E
3
运算符重载的方法
运算符重载的方法是定义一个重载运 算符的函数, 算符的函数,在需要执行被重载的运 算符时,系统就自动调用该函数, 算符时,系统就自动调用该函数,以 实现相应的运算。也就是说, 实现相应的运算。也就是说,运算符 重载是通过定义函数实现的。 重载是通过定义函数实现的。运算符 重载实质上是函数的重载。 重载实质上是函数的重载。
CCS E
21
运算符重载
如果将运算符重载函数作为成员函数, 如果将运算符重载函数作为成员函数, 它可以通过this this指针自由地访问本类 它可以通过this指针自由地访问本类 的数据成员, 的数据成员,因此可以少写一个函数 的参数。 的参数。但必须要求运算表达式第一 个参数(即运算符左侧的操作数) 个参数(即运算符左侧的操作数)是一 个类对象, 个类对象,而且与运算符函数的类型 相同。 相同。因为必须通过类的对象去调用 该类的成员函数,而且只有运算符重 该类的成员函数, 载函数返回值与该对象同类型, 载函数返回值与该对象同类型,运算 结果才有意义。 结果才有意义。
简述运算符重载的规则。
简述运算符重载的规则。
篇一:运算符重载是C/C++语言中一种强大的功能,允许程序员自定义函数的行为,以处理不同类型的数据。
运算符重载允许程序员在函数中重载算术、逻辑和位运算符,从而能够处理数组、结构体和指针等不同类型的数据。
以下是运算符重载的规则:1. 算术运算符重载算术运算符包括加号、减号、乘号和除号。
每个算术运算符都有一组默认的行为,可以通过运算符重载来自定义它们的行为。
例如,重载加号运算符可以使函数接受一个整数参数,并返回一个新的整数。
下面是一个简单的例子,演示了如何重载加号运算符:```c++struct MyStruct {int value;};MyStruct operator+(const MyStruct& other, int value) {return MyStruct(value + other.value);}int main() {MyStruct mystruct1 = { 10 };MyStruct mystruct2 = { 20 };int result = mystruct1 + mystruct2;std::cout << "result = " << result << std::endl;return 0;}```在上面的例子中,我们定义了一个名为`MyStruct`的结构体类型,其中包含一个整数类型。
然后,我们定义了一个重载加号运算符的函数,该函数接受一个整数类型的参数,并返回一个新的`MyStruct`对象。
在`main`函数中,我们定义了两个`MyStruct`对象`mystruct1`和`mystruct2`,并将它们相加,结果存储在`result`变量中。
2. 逻辑运算符重载逻辑运算符包括条件运算符和逻辑非运算符。
每个逻辑运算符都有一组默认的行为,可以通过运算符重载来自定义它们的行为。
运算符重载
运算符重载为类的友元函数
类中的声明: 类中的声明: friend 函数类型 operator 运算符(参数表); 运算符(参数表); 运算符重载函数的定义形式: 运算符重载函数的定义形式: 运算符(参数表) 函数类型 operator 运算符(参数表) { 函数体; 函数体; } 附加例) (附加例)用友元函数实现复数的加法运算符的重 载。
数据类型转换 标准数据类型转换
隐式转换: 自动转成double型) 隐式转换:7.5+2 (将2自动转成 将 自动转成 型 显式转换: 转成整型) 显式转换:int(89.5) (将89.5转成整型 将 转成型
涉及到类类型的转换
对于用户自己声明的类型, 对于用户自己声明的类型,编译系统并不知道怎样进行转 换。解决这个问题的关键是让编译系统知道怎样去进行转 换,需要定义专门的处理函数。 需要定义专门的处理函数。 将一个其他类型的数据转换成一个类类型: 将一个其他类型的数据转换成一个类类型:转换构造函数 将一个类的对象转换成另一类型的数据: 将一个类的对象转换成另一类型的数据:类型转换函数
运算符重载的格式
运算符重载为类的成员函数 运算符重载为类的友元函数
运算符重载的规则
①
② ③ ④ ⑤
.”、 *” ::”、 ?:”和 sizeof”等几个运 除“. 、“*”、“:: 、“?: 和“sizeof 等几个运 算符不能重载外, 算符不能重载外,C++中几乎所有的运算符都可以 中几乎所有的运算符都可以 重载。 重载。 运算符被重载后,其优先级和结合性不会改变。 运算符被重载后,其优先级和结合性不会改变。 不能改变运算符操作对象的个数。 不能改变运算符操作对象的个数。 运算符的意义应不变。 运算符的意义应不变。 用于类对象的运算符一般必须重载,但有两个例外, 用于类对象的运算符一般必须重载,但有两个例外, 运算符“ 和 不必用户重载。 运算符“=”和“&”不必用户重载。 不必用户重载
运算符重载——精选推荐
运算符重载运算符重载学习运算符重载,让运算符能做⼀些原来做不了的事情,⽅便它的使⽤⼀、运算符重载的概念1、什么是运算符重载1.重载:重新载⼊,就像之前学的函数重载,对⼀个已有的函数赋值⼀个新的定义,因此同⼀个函数名就可以有不同的含义。
2.运算符也是可以重载的,⽐如cout在输出⼀个变量的时候,能接受不同类型的数据并输出,他就是重载了<<运算符,这个就是运算符重载3.所以运算符重载指的是对已有的运算符重新定义新的运算规则,以适应不同的数据类型,当然重载之后之前的运算规则还是有的2、为什么要进⾏运算符重载1.运算符重载之后可以让运算符去适应不同的数据类型,对于基本数据类型,系统给出了运算符的操作规则,对于⾃定义数据类型来说,系统不知道该给出什么规则class student{int id;int age;char name[20];public:student(int id,int age,const char* name){this->id=id;this->age=age;strcpy(this->name,name);}}student stu1(1,23,"张三");student stu2(2,24,"李四");stu1+stu2;//如果是这样相加,那么应该加的是什么呢?编译器是不知道的,所以编译器就提供了运算符重载这个机制,让⽤户⾃定义运算符的运算规则⼆、运算符重载1、运算符重载类中定义1.关键字:operator,通过关键字来定义运算符重载(跟写个函数⼀样)2.定义:函数返回值类型 operator 要加载的运算符(参数列表){函数体;}这⾥把运算符的使⽤,理解为调⽤函数,只是和平时的调⽤函数有⼀点区别#include<iostream>#include<string>using namespace std;class student{int id;int age;string name;public:student(int age){this->age = age;id = 1;name = "sss ";}student(int id, int age, string name){this->id = id;this->age = age;this->name = name;}void showstudent(){cout << id << "\t" << age << "\t" << name << endl;}student operator+(student& p1)//这个函数会返回⼀个新的对象{int x=this->age + p1.age;student p(x);return p;//返回的是⼀个对象,会调⽤拷贝构造}int operator-(int x){return this->id - x;}void operator+(student&p2){cout << this->id + p2.id << endl;}};//1.操作这个运算符之后,返回值类型是什么int main(){student p1(0, 1, "yunfei");int x = p1.operator-(1);cout << x << endl;student stu1(1, 23, "张三");student stu2(2, 24, "李四");//student stu3 = stu1.operator+(stu2);//student stu3 = stu1 + stu2;stu1 + stu2;//stu3.showstudent();system("pause");return 0;}注意:因为我们这个运算符是在类中写的,所以是通过对象调⽤的,那么this指针会占⼀个参数,⽽且是第⼀个参数,也就是说我们重载⼀个运算符,是在类中,⽽这个运算符是个单⽬运算符,那么参数列表就不⽤写东西了,是双⽬运算符,那么就需要传另⼀个参数进来绝⼤部分的运算符重载都可以参照上⾯这个+号重载2、运算符重载的特点1.⼏乎所有的运算符都可以被重载,除了 . :: ()?() ) sizeof()2.运算符重载基本出现在类中和结构体中3.运算符可以理解为函数的⼀个表现3、运算符重载的注意事项1.重载运算符,这个重载的运算符还是满⾜原来的原则,但不能说重载+号,结果做的事-号的事,这样会使运算符的运⽤上增加很⼤的难度2.运算符重载的参数,类中重载调⽤对象会占⼀个参数,就是this会占⼀个参数,参数列表就是⽤来表⽰运算符的操作的3.对于运算符重载的调⽤,可以直接使⽤运算符,也可以通过对象 . 出来调⽤4.考虑返回值,不同的运算符有不同的返回值,要记得满⾜运算符原来的规则4、使⽤友元函数,实现运算符重载1.类在已经实现且部分修改的情况下下,需要进⾏运算符重载,就可以通过友元的⽅式来进⾏重载#include<iostream>#include<string>using namespace std;class person{int id;int age;string name;public:person(int id, int age, string name){this->id = id;this->age = age;this->name = name;}void showperson(){cout << id << "\t" << age << "\t" << name << endl;}friend int operator+(person&p1, person&p2);};//形参使⽤的是类对象的引⽤,在实参传对象的时候不会调⽤拷贝构造int operator+(person&p1, person&p2){return p1.id + p2.id;}//1.操作这个运算符之后,返回值类型是什么int main(){person stu1(1, 23, "张三");person stu2(2, 24, "李四");int x = operator+(stu1, stu2);//显⽰调⽤int y = stu1 + stu2;//隐式调⽤cout << x << endl << y << endl;system("pause");return 0;}容器:#include<iostream>#include<vector>using namespace std;int main(){//int 是v1这个容器存的类型vector<int> v1;for (int i = 0; i < 10; i++){//push_back()是⼀个函数,功能是尾插元素v1.push_back(i + 1);}for (int i = 0; i < 10; i++){cout << v1[i] << "\t";}system("pause");return 0;}左移右移运算符重载:#include<iostream>using namespace std;class person{int id;public:person(int id){this->id = id;}friend ostream& operator<<(ostream& os, person& p1);friend istream & operator>>(istream & in, person & p2);};//左移右移运算符重载,必须在类外重载,通过友元实现ostream& operator<<(ostream& os, person& p1)//左移运算符{os << p1.id << endl;return os;//返回的是⼀个cout,⽽且只能⽤引⽤}istream & operator>>(istream & in, person & p2)//右移运算符{in >> p2.id;return in;}int main(){person p1(10), p2(20);cin >> p1 >> p2;cout << p1 << endl << p2 << endl;system("pause");return 0;}前++,后++运算符重载:#include<iostream>using namespace std;class person{int id;public:person(int id){this->id = id;}person& operator++()//前++{this->id++;return *this;}person& operator++(int)//后++,int是⼀个占位符,⽤来区分前++和后++的{static person temp = *this;//引⽤不能返回局部变量,要⽤静态变量this->id++;return temp;}friend ostream& operator<<(ostream& os, person& p1);friend istream & operator>>(istream & in, person & p2);};//左移右移运算符重载,必须在类外重载,通过友元实现ostream& operator<<(ostream& os, person& p1)//左移运算符{os << p1.id << endl;return os;//返回的是⼀个cout,⽽且只能⽤引⽤}istream & operator>>(istream & in, person & p2)//右移运算符{in >> p2.id;return in;}int main(){person p1(10), p2(20);//cin >> p1 >> p2;//cout << p1 << endl << p2 << endl;cout << p1 ;//10cout << p1++ ;//10cout << p1 ;//11cout << ++p1 ;//12cout << p1 ;//12system("pause");return 0;}等号运算符重载:#include<iostream>using namespace std;class person{char* name;public:person(const char* name){this->name = new char[strlen(name) + 1];strcpy(this->name, name);}person& operator=(person&p1)//⽤不⽤引⽤传参,要看返回的对象会不会消失 {if (this->name != NULL){delete[]this->name;this->name = NULL;}this->name = new char[strlen() + 1];strcpy(this->name, );return *this;}void show(){cout << name << endl;}~person()//如果有申请函数,就要加上析构函数{if (name != NULL){delete[]name;name = NULL;}}};int main(){{person p1("张三"), p2("李四"), p3("王五");p1 = p2 = p3;p1.show();p2.show();p3.show();}//加上⼤括号,让对象死亡,就能调⽤析构函数system("pause");return 0;}智能指针和==号运算符重载:#include<iostream>using namespace std;class person{int id;public:person(int id){this->id = id;}void show(){cout << id << endl;}bool operator==(person& p){return this->id == p.id;}~person(){cout << "person的析构函数" << endl;}};class smartpointer{person* ps;//包含你要new出来的对象的类的指针public:smartpointer(person* p){ps = p;}//重载->person* operator->()//传回来的是地址,不是对象,不⽤引⽤{return ps;}//重载*person& operator*()//返回的是对象,会调⽤拷贝构造,所以⽤返回值⽤引⽤,就不会再调⽤拷贝构造了 {return *ps;//得到⼀个对象,}~smartpointer(){if (ps != NULL){delete ps;ps = NULL;}}};int main(){{smartpointer p(new person(5));p->show();(*p).show();person p1(1), p2(3);cout << (p1 == p2) << endl;}//有三个对象,所以析构函数执⾏了三次system("pause");return 0;}[]运算符重载:#include<iostream>using namespace std;class person{char* name;public:person(const char* name){this->name = new char[strlen(name) + 1];strcpy(this->name, name);}char& operator[](int index){return name[index];}~person(){if (name != NULL){delete[]name;name = NULL;}cout << "这是析构函数" << endl;}};int main(){person p("asdfg");cout << p[3] << endl;system("pause");return 0;}c++引⽤作为函数返回值:1.以引⽤返回函数值,定义函数时需要在函数名前加 &2.⽤引⽤返回⼀个函数值的最⼤好处是,在内存中不产⽣被返回值的副本3.返回值为引⽤的时候,返回的是⼀个地址,隐形指针4.当返回值不是引⽤时,编译器会专门给返回值分配出⼀块内存的引⽤作为返回值,必须遵守以下规则:(1)不能返回局部变量的引⽤。
第十三章运算符重载
void Counter::operator--( )
{ }
if(value>0)
value--;
unsigned int Counter:operator( )( )
{ }
return value;
main()
{
Counter mycounter; for(int i=0; i<12;i++)
{
的人民币对象的自增运算符。 编译解释为:
operator++(d3)调用RMB& operator++(RMB& S)进
行求值。
例: 按以下方式实现字符串对象的连接操作: stringream.h> #include <string.h> class String { public: String(int size=80) { length=size; buf=new char[length+1]; *buf=„\0‟; } String(char *s) { length=strlen(s); buf=new char[length+1]; strcpy(buf,s); }
注意:程序中重载运算符“++”、“--”和“( )”使用的操作数为 类对象;预定义运算符“++”、“--”使用的操作符为基本类型。
例如:下边是运算符重载的另一个例子,程序实现了复数的加 和减运算。
#include <iostream.h> class Complex { public: Complex() { real=imag=0.0; } Complex(double r) { real=r; imag=0.0; } Complex(double r,double i) { real=r; imag=i; } Complex operator+(const Complex& c); Complex operator-(const Complex& c); friend void print(const Complex& c); private: double real,imag; };
13 函数与运算符重载
使用“类名::”加以区分。 根据调用函数对象加以区分。
函数重载
范例 #include <iostream.h> class point { int x,y; public: point(int x,int y) { point::x=x; point::y=y; } float area() {return 0.0;} }; class circle:public point { int r; public: circle(int x,int y,int r1):point(x,y) { r=r1;} float area(){ return 3.14*r*r;} };
多态性概念
因此:
封装性是基础,用于实现类的定义; 继承性是关键,实现代码与数据的重用; 而多态性是补充,实现不同的行为表现,解决 程序的复杂性和易用性。 函数重载 运算符重载 动态联编
多态性的表现形式:
函数重载
C++中,当有一组函数完成相似功能时,函数名 允许重复使用,编译器根据参数表中参数的个数 或类型(不能根据形参变量名)来判断调用哪一 个函数,这就是函数的重载,且可以是成员函数 也可以是非成员函数。 重载函数仅仅靠函数的参数表,即参数的个数或 参数的类型进行区别。仅仅返回值不同的重载函 数是非法的。
高级语言(C++)程序设计
第15讲 函数与运算符重载
主要内容(2学时)
多态性概念 函数重载 运算符重载
多态性概念
面向对象中的多态性指系统发出同样的消 息被不同类型的对象接受时导致不同的行 为。这里的消息可理解为成员函数的调用, 而不同行为是指不同的函数实现。
运算符重载
万象城春节营销方案万象城春节营销方案一、背景介绍春节是中国传统的重要节日之一,也是全年销售额最高的时期之一。
随着消费水平的提高和消费观念的改变,人们在春节期间不仅满足于传统的购物需求,更加注重消费体验和品牌形象。
作为一家位于城市商圈的综合购物中心,万象城需要制定一套有吸引力、创新性的春节营销方案,提升品牌形象、增加销售额。
二、目标和定位1. 主要目标:增加春节期间销售额,提升品牌形象。
2. 定位:高端购物中心、高品质消费场所。
三、核心信息1. 品牌核心信息:高品质、多元、高端。
2. 营销信息:新年新气象,尽享购物乐趣。
四、理念和创意1. 理念:为顾客带来春节购物的新鲜元素,打造愉悦、独特的购物体验,让他们惊喜连连,享受购物乐趣。
2. 创意:结合传统与现代、东西方文化相融合,以独特的方式展现春节主题,吸引顾客的注意力。
五、活动策划1. 主题活动活动一:万象迎春晚会活动时间:春节前一天晚上活动地点:万象城中心广场活动内容:- 邀请知名主持人和明星嘉宾,举办一场充满欢乐、娱乐性的晚会。
演唱、舞蹈、小品、魔术等形式丰富多样。
- 晚会通过现场直播和社交媒体推广,吸引更多人关注和参与,增加品牌曝光。
- 晚会现场设置特色互动游戏,如抽奖、发红包等,增加顾客参与度,扩大品牌影响力。
活动二:文化体验展览活动时间:春节期间活动地点:万象城大堂及商铺内部活动内容:- 展览展示中国传统文化艺术,如中国书法、民俗文化、剪纸等。
- 安排专业讲解员,向顾客介绍传统文化背景和艺术品鉴赏技巧。
- 设置DIY手工坊,让顾客亲身参与传统艺术品制作,增加消费者的体验感。
活动三:花灯巡游活动时间:春节期间活动地点:万象城周边街区活动内容:- 邀请民间艺术团队表演传统舞蹈、武术,穿着中国传统服装,带着花灯游历街区。
- 设立观众席供顾客观看,并提供中国传统小吃、特色饮品等,让顾客在观看表演的同时,感受传统文化的美味。
2. 营销推广活动- 春节福袋:顾客在万象城购物满一定金额可以领取春节福袋,福袋内含有不同价值的现金券和特别优惠券,鼓励顾客消费,增加销售额。
运算符重载解读
02
在重载运算符时,应明确指定 运算符的参数类型和返回类型 ,避免与其他操作符混淆。
03
在重载运算符时,应尽量使用 有意义的操作符名称,以提高 代码的可读性。
考虑性能和效率
01
02
03
运算符重载时,应考虑 性能和效率,避免对原 有代码造成不必要的性
能损失。
在重载运算符时,应尽 量减少额外的计算和内 存开销,以提高代码的
一元运算符重载
一元运算符重载是指对一元运算符进行重载,使其能够应用于一个操作数。例如,在C中,我们可以 重载一元减号运算符(-)来返回一个数的相反数。
常见的一元运算符有:+、-、~、!等。
二元运算符重载
二元运算符重载是指对二元运算符进行重载,使其能够应用于两个操作数。例如,在C中,我们可以重载加号运算符(+)来定 义两个自定义类型的相加行为。
01
提高代码可读性
运算符重载可以使代码更易读, 因为运算符的使用可以使代码更 简洁、更直观。
02
03
方便函数调用
运算符重载可以使得函数调用更 加方便,因为运算符的使用可以 减少函数调用的开销。
04
缺点分析
易造成混淆
运算符重载可能会使得代码难以理解,因为 运算符的行为可能会被改变,导致读者难以
理解代码的含义。
常见的二元运算符有:+、-、*、/、%等。
三元运算符重载
三元运算符重载是指对三元运算符进行重载,使其能够应用于三个操作数。例如,在C中,我们可以重载条件运 算符(?)来定义一个自定义类型的条件判断行为。
常见的三元运算符有:?:。
03
运算符重载的常见场景
类与类之间的运算
01
两个类对象之间的加法运算,可以表示它们之间的某种关联或 合并。
运算符重载详解
运算符重载详解1.运算符重载定义:C++中预定义的运算符的操作对象只能是基本数据类型。
但实际上,对于许多⽤户⾃定义类型(例如类),也需要类似的运算操作。
这时就必须在C++中重新定义这些运算符,赋予已有运算符新的功能,使它能够⽤于特定类型执⾏特定的操作。
运算符重载的实质是函数重载,它提供了C++的可扩展性,也是C++最吸引⼈的特性之⼀。
运算符重载是通过创建运算符函数实现的,运算符函数定义了重载的运算符将要进⾏的操作。
运算符函数的定义与其他函数的定义类似,惟⼀的区别是运算符函数的函数名是由关键字operator和其后要重载的运算符符号构成的。
运算符函数定义的⼀般格式如下:<返回类型说明符> operator <运算符符号>(<参数表>){<函数体>} 2.运算符重载时要遵循以下规则:(1) 除了类属关系运算符"."、成员指针运算符".*"、作⽤域运算符"::"、sizeof运算符和三⽬运算符"?:"以外,C++中的所有运算符都可以重载。
(2) 重载运算符限制在C++语⾔中已有的运算符范围内的允许重载的运算符之中,不能创建新的运算符。
(3) 运算符重载实质上是函数重载,因此编译程序对运算符重载的选择,遵循函数重载的选择原则。
(4) 重载之后的运算符不能改变运算符的优先级和结合性,也不能改变运算符操作数的个数及语法结构。
(5) 运算符重载不能改变该运算符⽤于内部类型对象的含义。
它只能和⽤户⾃定义类型的对象⼀起使⽤,或者⽤于⽤户⾃定义类型的对象和内部类型的对象混合使⽤时。
(6) 运算符重载是针对新类型数据的实际需要对原有运算符进⾏的适当的改造,重载的功能应当与原有功能相类似,避免没有⽬的地使⽤重载运算符。
(7)重载运算符的函数不能有默认的参数,否则就改变了运算符的参数个数,与前⾯第3点相⽭盾了;(8)重载的运算符只能是⽤户⾃定义类型,否则就不是重载⽽是改变了现有的C++标准数据类型的运算符的规则了,会引会天下⼤乱的;(9)⽤户⾃定义类的运算符⼀般都必须重载后⽅可使⽤,但两个例外,运算符“=”和“&”不必⽤户重载;(10)运算符重载可以通过成员函数的形式,也可是通过友元函数,⾮成员⾮友元的普通函数。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
杨章伟 e-mail:yang505412@
融智技术学院
课程内容安排
• • • • • •
运算符重载简介 运算符重载的形式 特殊运算符的重载 类类型转换 小结 习题
融智技术学院
1-2
运算符重载简介
• 第12章介绍了多态的实现技术,主要包括函数的重载和虚函 数。事实上,本章介绍的运算符重载也是多态性实现的一个重 要手段。运算符重载实现的是编译时的多态,即静态多态性。 C++预定义的运算符只是对基本数据类型进行操作,而对于自 定义的数据类型比如类,却没有类似的操作。为了实现对自定 义类型的操作,就必须自己编写程序来说明某个运算符作用在 这些数据类型上时,应该完成怎样的操作,这就要引入运算符 重载的概念。 简单地说,运算符重载是实现编译时多态性的另外一种形式。 运算符重载是对已有的运算符赋予多重含义,使同一个运算符 作用于不同类型的数据时,实现不同类型的行为。
融智技术学院
1-14
融智技术学院
1-6
• • • • • •
“++”和“--”的重载
• 前面范例中介绍了“++”和“--”的重载,但运算符“++”和“-”有前置和后置两种形式。例如,表达式“a++”和表达式 “++a”是不一样的。 如果不区分前置和后置,则使用operator++()或operator--()即 可;否则,要使用operator++()或operator--()来重载前置运算 符,使用operator++(int)或operator--(int)来重载后置运算符 ,调用时,参数int被传递给值0。 【范例13-6】“++”和“--”运算符的重载。范例实现了运算符 “++”和“--”的重载,包括其前置和后置两种形式
融智技术学院
1-8
• • • • •
下标运算符“[ ]”的重载
• 下标运算符operator[]通常用来访问数组中的某个元素。 事实上,其可以看作是一个双目运算符,第一个运算符是 数组名,第二个运算符是数组下标。在类对象中,可以重 载下标运算符,用它来定义相应对象的下标运算。
【范例13-8】下标运算符“[ ]”的重载。该范例实现了下 标运算符“[ ]”的重载,重载后的下标运算符能够判断数 组的下标是否越界,如越界则给出错误信息
1-11
小结
本章主要介绍了运算符重载的相关内容。从运算符重载的定 义和特点着手,通过一个具体的范例介绍运算符重载的 优点,接着简要说明了运算符重载的规则。同时,重点 讲解了两种形式的运算符重载:运算符重载为成员函数 和运算符重载为友元函数。针对每一种重载方式,都通 过一个范例介绍了双目运算符的重载和单目运算符的重 载。此外,针对几种特殊运算符,本章通过实例讲解了 赋值运算符、下标运算符等重载的实现。最后,就C++面 向对象程序设计的基础概念和特征做了简要的概括。
•
融智技术学院
1-9
类类型转换
类类型是指某个对象的数据类型为类,而不是标准的数据类 型。在C++中,标准的数据类型与类类型之间的转换有三 种方法。 通过构造函数转换:通过构造函数能将标准数据类型向类类 型转换,但不能将类类型转换为标准类型。 通过类类型转换函数转换:要将类类型转换为标准数据类型 时,需要采用显式类型转换机制,定义类类型转换函数 。 通过运算符重载实现类型转换:可以实现标准类型的数据与 类对象之间的运算。
融智技术学院
1-3
•
运算符重载的定义
运算符重载的实质是函数重载。事实上,C++语言中的每一个运算符 对应着一个运算符函数,在实现过程中,把指定的运算表达式中 的运算符转化为对运算符函数的调用,而表达式中的运算对象转 化为运算符函数的实参,这个过程是在编译阶段完成的。例如: int a=1,b=2; a+b; 表达式“a+b”在编译前,将被解释为函数调用形式:operator+(a,b) 。其中,operator是一个关键字,它与后面的“+”共同组成了该 运算符函数的函数名。因此,可以将运算符重载看作是一种特殊 的函数重载。运算符函数定义的一般格式如下: <返回类型说明符> operator <运算符符号>(<参数表>) { <函数体> }
融智技术学院
1-10
【范例13-9】运算符重载转换运算符
C++中定义一个类的类型转换函数的形式为: <类名>::operator type() { // return type类型的数据 //返回type类型的对象 } 范例在类中重载了一个类型转换函数int(),该函数实现输出 并返回类类型值
融智技术学院
融智技术学院
1-5
运算符重载的形式
• 运算符的重载形式有两种:一种是重载为类的成员函数, 一种是重载为类的友元函数。对于每一种重载形式,由于 运算符的不同,都可以将其主要分为双目运算符和单目运 算符的实现。 重载为类的成员函数 : 双目运算符重载为成员函数 单目运算符重载为成员函数 重载为类的友元函数 : 双目运算符重载为友元函数 单目运算符重载为友元函数
融智技术学院
1-12
习题
• • 1.定义计数器Counter类,对其重载运算符 +。 【解答】该试题主要考查双目运算符的重载实现。运算 符的重载可以在类中进行定义,也可以在类外进行定义 ,但在类中必须进行说明。该试题首先定义一个计数器 类,在公有成员中说明运算符的重载,在类外进行定义 。 2.对Point类重载++(自增)、--(自减)运算符。 【解答】该试题主要考查单目运算符++(自增)、--(自 减)的重载。读者需要注意,++(自增)、--(自减)均 具有前置和后置的区别,其属于特殊运算符的重载。因 此,在进行这两种运算符的重载时,需要考虑前置和后 置。该试题首先定义类Point,在类中对运算符重载函数 进行说明,在类外对重载函数进行具体定则
简单地说,实现运算符重载即编写一个函数,该函数以 “operator运算符号”为函数名,其定义了重载的运算符 将要执行的操作。此外,函数的形参类型必须是自定义的类 型。当使用该运算符对形参规定的数据类型进行运算时,就 执行函数体中的操作,而覆盖了原运算符的功能。 C++中的运算符除了几个不能重载外,其他的都能重载,而且 只能重载已有的运算符,不能自己另编。 重载以后运算符的优先级和结合性都不能改变,语法结构也不 能改变,即单目运算符只能重载为单目运算符,多目运算符 只能重载为多目运算符。 运算符重载以后的功能应与原有功能类似,含义必须清楚,不 能有二义性。
• •
融智技术学院
1-13
习题
4.设计一个程序,重载四则运算符,实现复数的算术运算 。 【解答】该试题主要考查算术运算符的重载。复数由实部和 虚部构造,因此可以定义一个复数类,然后再在类中重 载复数四则运算的运算符。具体的说,可以定义类 complex,其中将四则运算符“+”、“-”、“×”和“/” 都重载为类的成员函数,重载后的四则运算符能够实现 复数的四则运算。同时声明一个友元函数用于输出运算 结果。在主函数main()中创建三个对象,其中两个对象 进行四则运算。
•
•
融智技术学院
1-7
赋值运算符“=”的重载
• 除了前面讲解的算术运算符可以进行重载外,许多其他 类型的运算符也可以进行重载,赋值运算符“=”也是如 此。在进行赋值运算符“=”重载的应用前,读者应先了 解赋值运算符“=”在实际程序中的运行情况。事实上, 对于任何一个类,如果没有用户自定义的赋值运算符函 数,系统会自动地为其生成一个默认的赋值运算符函数 ,以完成数据成员之间的复制。例如,下面的程序段: X & X::operator=(const X &source) { //类对象成员之间的赋值语句 } 一旦类X的两个对象ob1和ob2已创建,就可用 ob1=ob2进行赋值了。