虚函数与多态知识总结

转眼间就到学期末了,这是本学期最后一堂课了......

一、知识总结

(一)多态

多态性(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、虚函数的重载特性

在派生类中重载基类的虚函数要求函数名、返回类型、参数个数、参数类型和顺序完全相同, 如果仅仅返回类型不同,C++认为是错误重载, 如果函数原型不同,仅函数名相同,丢失虚特性 
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++的学习之路。回收当初啥都不会的那段时间,还是挺怀念的,现在虽然算不上学的很好,但基本知识还是掌握了的。接下来就是复习阶段了。个人感觉多态用的情况会不如继承多,多态就像是一个菜单样式的东西。好了,继续努力吧!





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值