C++ 在继承中虚函数、纯虚函数、普通函数,三者的区别

1.虚函数(impure virtual)

  C++的虚函数主要作用是“运行时多态”,父类中提供虚函数的实现,为子类提供默认的函数实现

  子类可以重写父类的虚函数实现子类的特殊化。

  如下就是一个父类中的虚函数:

复制代码
class A
{
public:
    virtual void out2(string s)
    {
        cout<<"A(out2):"<<s<<endl;
    }
};
复制代码

2.纯虚函数(pure virtual)

  C++中包含纯虚函数的类,被称为是“抽象类”。抽象类不能使用new出对象,只有实现了这个纯虚函数的子类才能new出对象。

  C++中的纯虚函数更像是“只提供申明,没有实现”,是对子类的约束,是“接口继承”。

  C++中的纯虚函数也是一种“运行时多态”。

  如下面的类包含纯虚函数,就是“抽象类”:

复制代码
class A
{
public:
    virtual void out1(string s)=0;
    virtual void out2(string s)
    {
        cout<<"A(out2):"<<s<<endl;
    }
};
复制代码

3.普通函数(no-virtual)

  普通函数是静态编译的,没有运行时多态,只会根据指针或引用的“字面值”类对象,调用自己的普通函数

  普通函数是父类为子类提供的“强制实现”。

  因此,在继承关系中,子类不应该重写父类的普通函数,因为函数的调用至于类对象的字面值有关。

4.程序综合实例

复制代码
#include <iostream>
using namespace std;

class A
{
public:
    virtual void out1()=0;  ///由子类实现
    virtual ~A(){};
    virtual void out2() ///默认实现
    {
        cout<<"A(out2)"<<endl;
    }
    void out3() ///强制实现
    {
        cout<<"A(out3)"<<endl;
    }
};

class B:public A
{
public:
    virtual ~B(){};
    void out1()
    {
        cout<<"B(out1)"<<endl;
    }
    void out2()
    {
        cout<<"B(out2)"<<endl;
    }
    void out3()
    {
        cout<<"B(out3)"<<endl;
    }
};

int main()
{
    A *ab=new B;
    ab->out1();
    ab->out2();
    ab->out3();
    cout<<"************************"<<endl;
    B *bb=new B;
    bb->out1();
    bb->out2();
    bb->out3();

    delete ab;
    delete bb;
    return 0;
}
复制代码

执行结果:

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
虚函数纯虚函数都是C++语言面向对象的概念,它们的作用是支持多态性和动态绑定。但是,虚函数纯虚函数之间还是有一些区别的。 1. 虚函数是一个可以被继承类重写的函数,而纯虚函数是一个没有实现的函数,必须由继承类实现。 2. 虚函数可以有实现,也可以没有实现,而纯虚函数必须没有实现。如果一个类纯虚函数,那么这个类就是抽象类,不能被实例化。 3. 如果一个派生类没有实现其基类纯虚函数,那么它也是一个抽象类,不能被实例化。 4. 如果一个类虚函数,那么这个类的对象会包含一个虚函数表指针,虚函数存储着虚函数的地址。在运行时,程序会根据对象的类型来确定调用哪个虚函数,这就是动态绑定。而纯虚函数也会在虚函数有一个条目,但是没有实现,需要由继承类实现。 下面是虚函数纯虚函数的示例代码: ```c++ class Animal { public: virtual void MakeSound() { // 基类虚函数 cout << "Animal makes sound!" << endl; } virtual void Eat() = 0; // 基类纯虚函数 }; class Cat : public Animal { public: virtual void MakeSound() { // 派生类虚函数 cout << "Cat meows!" << endl; } virtual void Eat() { // 派生类实现的纯虚函数 cout << "Cat eats fish!" << endl; } }; int main() { Animal* pAnimal = new Cat(); // 基类指针指向派生类对象 pAnimal->MakeSound(); // 调用派生类的虚函数 pAnimal->Eat(); // 调用派生类实现的纯虚函数 return 0; } ``` 在上面的示例代码,Animal是一个抽象基类,其有一个虚函数MakeSound和一个纯虚函数Eat。Cat是Animal的一个派生类,重写了MakeSound虚函数并实现了Eat纯虚函数。在main函数,通过基类指针指向派生类对象,实现了多态性和动态绑定。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值