转眼间就到学期末了,这是本学期最后一堂课了......
一、知识总结
(一)多态
多态性(Polymorphism)是指一个名字,多种语义;或界面相同,多种实现。重载函数是多态性的一种简单形式。虚函数允许函数调用与函数体的联系在运行时才进行,称为动态联编。
(二)虚函数和动态联编
冠以关键字virtual的成员函数称为虚函数。实现运行时多态的关键首先是要说明虚函数,另外,必须用基类指针调用派生类的不同实现版本
1、虚函数和基类指针
基类指针虽然获取派生类对象地址,却只能访问派生类从基类继承的成员
看如下代码:
#include<iostream>
usingnamespace std;
class Base
{public : Base(char xx) { x = xx; }
void who() { cout<< "Base class: " << x << "\n" ; }
protected: char x;
};
class First_d: public Base
{public : First_d(charxx, char yy):Base(xx) { y = yy;}
void who() { cout<< "First derived class: "<< x << ", "<< y << "\n" ; }
protected: char y;
};
class Second_d: public First_d
{public :
Second_d(char xx, char yy,char zz) : First_d(xx, yy) { z = zz;}
void who() { cout<< "Second derived class: "<< x << ", "<< y << ", " << z << "\n" ; }
protected: char z;
};
intmain()
{Base B_obj('A' ) ; First_dF_obj('T', 'O' ) ; Second_dS_obj('E', 'N', 'D' ) ;
Base * p ;
p = & B_obj; p -> who() ;
p = &F_obj; p -> who() ;
p = &S_obj; p -> who() ;
F_obj.who();
( ( Second_d* ) p ) -> who() ;
}
需要注意
一个虚函数,在派生类层界面相同的重载函数都保持虚特性:虚函数必须是类的成员函数,不能将友元说明为虚函数,但虚函数可以是另一个类的友元,析构函数可以是虚函数,但构造函数不能是虚函数
2、虚函数的重载特性
class base
{ public :
virtual void vf1 ( ) ;
virtual void vf2 ( ) ;
virtual void vf3 ( ) ;
void f ( ) ;
} ;
class derived : public base
{ public :
void vf1 ( ) ; // 虚函数
void vf2 ( int ) ; // 重载,参数不同,虚特性丢失
char vf3 ( ) ; // error,仅返回类型不同
void f ( ) ; // 非虚函数重载
} ;
void g ( )
{ derived d ;
base * bp = & d ; // 基类指针指向派生类对象
bp -> vf1 ( ) ; // 调用 deriver :: vf1 ( )
bp -> vf2 ( ) ; // 调用 base :: vf2 ( )
bp -> f ( ) ; // 调用 base :: f ( )
} ;
3.虚析构函数
构造函数不能是虚函数。建立一个派生类对象时,必须从类层次的根开始,沿着继承路径逐个调用基类的构造函数;析构函数可以是虚的。虚析构函数用于指引 delete运算符正确析构动态对象
#include<iostream>
usingnamespace std ;
classA
{ public:
~A(){ cout<< "A::~A() is called.\n" ; }
} ;
classB : public A
{ public:
~B(){ cout<< "B::~B() is called.\n" ; }
};
intmain() {
A *Ap = new B ;
B *Bp2 = new B ;
cout<< "delete first object:\n" ;
delete Ap;
cout<< "delete second object:\n" ;
delete Bp2 ;
}
说明:
1.派生类应该从它的基类公有派生。?
2.必须首先在基类中定义虚函数。
3.派生类对基类中声明虚函数重新定义时,关键字virtual可以不写。
4.一般通过基类指针访问虚函数时才能体现多态性。
5.一个虚函数无论被继承多少次,保持其虚函数特性。
6.虚函数必须是其所在类的成员函数,而不能是友元函数,也不能是静态函数。
7.构造函数、内联成员函数、静态成员函数不能是虚函数。
(虚函数不能以内联的方式进行处理)
8.析构函数可以是虚函数,通常声明为虚函数。
(四)纯虚函数和抽象类
纯虚函数是一种特殊的虚函数,在许多情况下,在基类中不能对虚函数给出有意义的实现,而把它声明为纯虚函数,它的实现留给该基类的派生类去做。
这就是纯虚函数的作用。纯虚函数是一个在基类中说明的虚函数,在基类中没有定义,要求任何派生类都定义自己的版本,纯虚函数为各派生类提供一个公共界面
纯虚函数说明形式:
virtual 类型 函数名(参数表)=0;
二、例题引入
接下来是一个例题,简单的图表类
class figure
{ protected : double x,y;
public: void set_dim(double i, double j=0) { x = i ; y = j ; }
virtual void show_area() = 0 ;
};
class triangle : public figure
{ public :
void show_area()
{ cout<<"Triangle with high "<<x<<" and base "<<y <<" has an area of "<<x*0.5*y<<"\n"; }
};
class square : public figure
{ public:
void show_area()
{ cout<<"Square with dimension "<<x<<"*"<<y <<" has an area of "<<x*y<<"\n"; }
};
class circle : public figure
{ public:
void show_area()
{ cout<<"Circle with radius "<<x;
cout<<" has an area of "<<3.14*x*x<<"\n";
}
};
三、心得体会
本学期最后一堂课结束了,我们的大一生活也就快结束了,但学习之路不会中断,尤其是C++的学习之路。回收当初啥都不会的那段时间,还是挺怀念的,现在虽然算不上学的很好,但基本知识还是掌握了的。接下来就是复习阶段了。个人感觉多态用的情况会不如继承多,多态就像是一个菜单样式的东西。好了,继续努力吧!