第五章多态性

2018-05-29  本文已影响0人  割麦子洁

第5章  多态性

一、编译时的多态性和运行时的多态性

多态性是指用一个名字定义不同的函数,这些函数执行不同但又类似的操作,这样就可以用同一个函数名调用不同内容的函数。 也是指不同的对象收到相同的的消息时, 执行不同的操作

多态性包括编译时的多态性运行时的多态性

(1)编译时的多态性:静态联编支持的多态性称为编译时的多态性,也称静态多态性。

静态联编----静态联编是指系统在[编译时]就决定如何实现某一动作。静态联编要求在程序编译时就知道调用函数的全部信息。通过函数重载(运算符重载)和模板

 补充:static是静态变量,就是变量值不随函数执行结束而消失,下次调用同一函数时,上次所赋予的值仍存在。const是常量,是不可改变的。

(2)运行时的多态性:动态联编所支持的多态性称为运行时多态性,也称动态多态性。

动态联编----动态联编是指系统在运行时动态实现某一动作。一直要到[程序运行时]才能确定调用哪个函数。通过虚函数

补充:程序是【先自上而下编译再是运行】 


二、运算符函数

1、 在类外定义的运算符重载函数

运算符函数:用于完成“+” “-” “*” “/” 等操作的函数

运算符函数名:operator@(@指要重载的运算符),例如operator+,operator-,operator*

运算符重载:两个或两个以上的运算符函数共有同一个运算符函数名


(重载分为一般函数重载和运算符重载)

补充:函数的重载的规则:  1、函数名称必须相同。 2、参数列表必须不同(个数不同、类型不同、参数排列顺序不同等)。 3、函数的返回类型可以相同也可以不相同。 4、仅仅返回类型不同不足以成为函数的重载。


引发的问题:C++系统预先编好的运算符函数不能完成将两个对象的相加工作。

解决方法如下:调用运算符函数将两个对象相加

complex operator+(complexco1,complex co2) {

    complex temp;

    temp.real=co1.real+co2.real;

    temp.imag=co1.imag+co2.imag;

    return temp;

}

[调用]运算符函数operator+()的有2种方式:

第一种:total1=operator+(com1,com2);//显示调用

第二种:total2=com1+com2;//隐式(习惯)调用


注意:C++语言对运算符重载制定了以下一些规则:【重点】

(1)以下运算符不能重载. 成员访问运算符  .* 成员指针访问运算符  ::作用域运算符 sizeof 长度运算符 ?:条件运算符

(2)不允许用户自己定义新的运算符进行重载

(3)最好保持运算符原含义,容易被人接受

(4) 重载不能改变运算符的操作对象(即操作数)的个数

(5) 重载不能改变运算符原有的优先级

例如,先乘除,后加减,因此表达式:x=y-a*b;等价于  x=y-(a*b);

 (5) 重载不能改变运算符原有的结合特性

例如:乘、除法运算符“*”和“/”都是左结合的

 (6) 运算符重载函数的参数至少应有一个类对象(或类对象的引用)

    例如以下定义运算符重载函数的方法是错误的:

       int operate+(int x,int y)【标准运算符】

       { return x+y;}

 (7)赋值运算符“=”,不必类对象用户进行重载

引发问题:运算符函数只能访问公有数据成员,而【不能访问类的私有数据成员】问题

解决方法:定义为它将要操作的类的成员函数(简称为成员运算符重载函数);

                   或者定义为它将要操作的类的友元函数(简称为友元运算符重载函数)

2、友元运算符重载函数定义:

(1)在类的内部定义友元函数格式为:

(2)在类中声明友元函数的原型,在类外定义

 friend complex operator+(complex co1,complex co2);

complex operator+(complex co1,complex co2)

补充:其他函数可以按照先声明后使用的方式,在main函数前先加一句声明;也可以直接调用,把函数放在main函数前。

注:在友元运算符重载函数的形参表中

若重载的是双目运算符,则参数表中有两个操作;+ *

若重载的是单目运算符,则参数表中只有一个操作。- & ! ++


说明:

(1)在函数返回的时候,可以直接用类的构造函数来生成一个临时对象,而不对该对象进行命名

Complex operator +(Complex& a,Complex& b){

         return Complex(a.real+b.real, a.imag+b.imag); 

        //临时也是无名对象,返回临时对象

}

引发的问题:使用友元函数重载“++”、“--”单目运算符,可能出现一些问题,会导致对形参的修改无法传到函数体外,于是要采用传址调用,把对象的引用作为参数。

friend coord operator ++(coord& op);    // 正确

friend complex operator +(complex a,complex b)//正确

引起错误的原因是在于友元函数没有this指针,所以不能引用this指针所指的对象。若函数采用对象参数,通过传值的方法传递参数,函数体内对op的所有修改都无法传道函数体外。 

说明:

(1)运算符重载函数operator @可以返回任何类型,甚至可以是void类型,但通常返回类型与它所操作的类的类型相同,这样可以使重载运算符用在复杂的表达式中。

(2) 有的运算符不能定义为友元运算符重载函数,如赋值运算符=、下标运算符[]、函数调用运算符()等。

3.成员运算符重载函数【成员运算符函数的参数跟友元运算符函数不一样,比友元运算符函数少一个参数】

(1)在类的内部,定义成员运算符重载函数

(2)在类的内部, 声明成员运算符重载函数原型的格式如下:

注:在成员运算符重载函数的形参表中

若运算符是单目的,则参数表为空;

若运算符是双目的,则参数表中有一个操作数----右操作数,当前对象----左操作数,它是通过this指针隐含地传递给函数的。

4.成员运算符重载函数与友元运算符重载函数的比较【重点】

1. 对双目运算符而言,成员运算符重载函数带有一个参数,而友元运算符重载函数带有两个参数;对单目运算符而言,成员运算符重载函数不带参数,而友元运算符重载函数带一个参数。

2. 双目运算符一般可以被重载为友元运算符函数或成员运算符函数

  [  但有一种情况,必须使用友元函数。]

引发问题:上例中整数50不能调用成员运算符函数

解决方法:使用两个友元函数

                  friend nclass operator+(nclass ob,int x);

                  friend nclass operator+(int x,nclass ob);

一般而言:

·对于单目运算符++,建议选择成员函数;

·对于运算符“=、()、[]、->”只能作为成员函数;

·对于运算符“+=、-=、/=、*=、&=、!= 、~=、%=、<<=、>>=”,建议重载为成员函数;

·对于其他运算符,建议重载为友元函数。


5. + +和- -的重载

对于前缀方式 ++ob

1. 成员运算符函数为:ob.operator ++( );

2. 友元运算符函数为: operator ++(X& ob);//ob为类X对象的引用

对于后缀方式 ob++

1. 成员运算符函数为:ob.operator ++(int);// 增加一个参数int,表示与后缀方式

2. 友元运算符函数为: operator++(X &ob,int);// 调用时,参数int一般被传递给值0

6.  赋值运算符=的重载

调用默认的赋值运算符函数,将对象obj2的数据内容逐域拷贝到对象obj1中。


三、类型转换

1、两种类型转换方式

隐式转换:小化大

显示转化

2、转换构造函数也是一种构造函数,它只有一个参数,作用是将一个其他类型的数据转换成它所在类的对象【与构造函数要区别开来】

3、类型转换函数,作用是将一个类的对象转换成另一个类型的数据

说明:

1.类型转换函数只能定义为一个类的成员函数而不是定义为友元函数

2.没有参数,也没有在函数面前指定类型

3.必须有return

4.一个类可以有多个类型转换函数


五、虚函数

上一篇 下一篇

猜你喜欢

热点阅读