虚函数、纯虚函数、重载函数的作用与区别

 虚函数为了重载和多态的需要,在基类中是有定义的,即便定义是空,所以子类中可以重写也可以不写基类中的此函数!

纯虚函数在基类中是没有定义的,必须在子类中加以实现,很像java中的接口函数!

虚函数

引入原因:为了方便使用多态特性,我们常常需要在基类中定义虚函数。

class Cman

{

public:

virtual void Eat(){……};

void Move();

private:

};

class CChild : public CMan

{

public:

virtual void Eat(){……};

private:

};

CMan m_man;

CChild m_child;

CMan *p ;//这才是使用的精髓,如果不定义基类的指针去使用,没有太大的意义

p = &m_man ;

p->Eat(); //始终调用CManEat成员函数,不会调用 CChild 

p = &m_child;

p->Eat(); //如果子类实现(覆盖)了该方法,则始终调用CChildEat函数

//不会调用CMan  Eat 方法;如果子类没有实现该函数,则调用CManEat函数

p->Move(); //子类中没有该成员函数,所以调用的是基类中的

纯虚函数

引入原因:

1、同虚函数

2、在很多情况下,基类本身生成对象是不合情理的。例如,动物作为一个基类可以派生出老虎、孔雀等子类,但动物本身生成对象明显不合常理。

纯虚函数就是基类只定义了函数体,没有实现过程,定义方法如: virtual void Eat() = 0; 不要cpp中定义;纯虚函数相当于接口,不能直接实例话,需要派生类来实现函数定义;

有的人可能在想,定义这些有什么用啊 ,我觉得很有用,比如你想描述一些事物的属性给别人,而自己不想去实现,就可以定义为纯虚函数。说的再透彻一些。比如盖楼房,你是老板,你给建筑公司描述清楚你的楼房的特性,多少层,楼顶要有个花园什么的建筑公司就可以按照你的方法去实现了,如果你不说清楚这些,可能建筑公司不太了解你需要楼房的特性。用纯需函数就可以很好的分工合作了

虚函数和纯虚函数区别

观点一:

类里声明为虚函数的话,这个函数是实现的,哪怕是空实现,它的作用就是为了能让这个函数在它的子类里面可以被重载,这样的话,这样编译器就可以使用后期绑定来达到多态了

纯虚函数只是一个接口,是个函数的声明而已,它要留到子类里去实现。

class A{

protected:

void foo();//普通类函数

virtual void foo1();//虚函数

virtual void foo2() = 0;//纯虚函数

}

观点二:

虚函数在子类里面也可以不重载的;但纯虚必须在子类去实现,这就像Java的接口一样。通常我们把很多函数加上virtual,是一个好的习惯,虽然牺牲了一些性能,但是增加了面向对象的多态性,因为你很难预料到父类里面的这个函数不在子类里面不去修改它的实现

观点三:

虚函数的类用于实作继承,继承接口的同时也继承了父类的实现。当然我们也可以完成自己的实现。纯虚函数的类用于介面继承,主要用于通信协议方面。关注的是接口的统一性,实现由子类完成。一般来说,介面类中只有纯虚函数的。

观点四:

带纯虚函数的类叫虚基类,这种基类不能直接生成对象,而只有被继承,并重写其虚函数后,才能使用。这样的类也叫抽象类。

虚函数是为了继承接口和默认行为

纯虚函数只是继承接口,行为必须重新定义


虚函数与重载函数的区别 

                
                
    重载函数在类型和参数数量上一定不相同,而重定义的虚函数则要求参数的类型和个数、函数返回类型相同; 
    虚函数必须是类的成员函数,重载的函数则不一定是这样; 
    构造函数可以重载,但不能是虚函数,析构函数可以是虚函数。 
                        
                        

例1:

                               
  #include   <  iostream.h  > 
  class   CBase
  {
public:
    
virtual int func(unsigned char ch) {return --ch;}
}
  ;
  class   CDerive :   public   CBase
  {
    
int func(char ch) {return ++ch;}   //此为函数重载
}
  ;
  void   main()
  {    CBase *p=new CDerive;
    
int n=p->func(40);        //调用基类的 func()
    cout<<" the result is : "<<n<<endl;
}
                     
运行结果:
       the result is : 39
                       
                  

例2:

                 
  #include   <  iostream.h  > 
  class   CBase
  {
public:
    
virtual int func(unsigned char ch) {return --ch;}
}
  ;
  class   CDerive :   public   CBase
  {
    
int func(unsigned char ch) {return ++ch;}      //此为虚函数
}
  ;
  void   main()
  {    CBase *p=new CDerive;
    
int n=p->func(40);        //调用派生类的 func()
    cout<<" the result is : "<<n<<endl;
}
                
               
运行结果:
        the result is : 41

原文链接: http://blog.csdn.net/livelylittlefish/article/details/2171515
标签: <无>
    分享到 
    0
      0 |    0


    评论
    添加红包

    请填写红包祝福语或标题

    红包个数最小为10个

    红包金额最低5元

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

    抵扣说明:

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

    余额充值