虚函数与纯虚函数

简介

虚函数

在C++中虚函数是实现多态的一种机制,核心理念就是通过基类访问派生类定义的函数。
最常见的例子就是如下

class A
{
public:
    virtual void foo()
    {
        cout << "A::foo() is called" << endl;
    }
};
class B :public A
{
public:
    virtual void foo()
    {
        cout << "B::foo() is called" << endl;
    }
};
int main(void)
{
    A *a = new B();
    a->foo();
    return 0;
}

这里函数foo通过关键字virtual声明为了虚函数,主函数中定义了一个父类A的指针a,指向一个子类B,此时再用a去调用foo函数,调用的则是子类B中的foo函数。如果没有virtual关键字,那么上述过程中的foo函数就会调用A类中的构造函数。
一个类函数的调用并不是在编译时刻被确定的,而是在运行时刻被确定的这就叫做动态编联。由于编写代码的时候并不能确定被调用的是基类的函数还是哪个派生类的函数,所以被称为“虚”函数。

纯虚函数和抽象类

纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法。在基类中实现纯虚函数的方法是在函数原型后加“=0”。
含有纯虚函数的类被称为抽象类,不能从抽象类中实例化对象出来。那么抽象类的作用是什么呢?
当我们以动物做一个基类,从这个基类中可以派生出老虎,狮子等动物,但是从动物基类中抽象出一个实例出来显然是不合理的。为了处理这类问题,便引入了纯虚函数,这样使派生类中必须重写基类中的纯虚函数,也就继承了基类的特性。

注意

1.当基类对象要对派生类对象进行操作的时候,析构函数在基类中设置为虚函数,为了防止析构时只析构基类而没有析构派生类。例如:

class A
{
public:
    A() 
    {
         ptra_ = new char[10];
    }
    ~A()
    { 
         delete[] ptra_;
    } 
private:
    char * ptra_;
};

class B: public A
{
public:
    B()
    {
         ptrb_ = new char[20];
    }
    ~B()
    {
         delete[] ptrb_;
    }
private:
    char * ptrb_;
};
void foo()
{
A * a = new B;
delete a;
}

在这个例子中,程序也许不会象你想象的那样运行,在执行delete a的时候,实际上只有A::~A()被调用了,而B类的析构函数并没有被调用!这是否有点儿可怕?
如果将上面A::~A()改为virtual,就可以保证B::~B()也在delete a的时候被调用了。因此基类的析构函数都必须是virtual的。

2.构造函数任何情况下都不能设置为虚函数。因为虚函数的调用要在类的虚函数表中查找。而虚函数表是在构造函数中生成(准确来说是在初始化列表之前,所以构造函数中可以调用虚函数),如果构造函数为虚,就需要通过 虚函数表来调用,可是对象还没有实例化,也就是内存空间还没有,无法找到虚函数表,所以构造函数不能是虚函数。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中的虚函数是通过在基类中使用关键字virtual来声明的函数。虚函数可以在派生类中被重写,使得在运行时根据对象的类型动态调用正确的函数。虚函数的调用是基于对象的动态类型实现的,这意味着即使使用基类的指针或引用来调用虚函数,实际调用的仍然是对象的派生类中的函数。 纯虚函数是在基类中声明但没有实现的虚函数纯虚函数的声明以 "= 0" 结尾,表示该函数没有函数体。纯虚函数的存在使得基类成为抽象类,抽象类不能被实例化。派生类必须实现基类中的纯虚函数才能被实例化。 构造函数不能被声明为虚函数,因为虚函数的调用依赖于对象的类型,而在构造函数中对象的类型尚未确定。因此,构造函数无法使用虚函数的动态派发机制。 析构函数可以被声明为虚函数,当基类指针指向派生类对象并通过该指针删除对象时,如果析构函数不是虚函数,那么只会调用基类的析构函数而不会调用派生类的析构函数,从而导致派生类对象没有正确地被销毁。使用虚析构函数可以确保在删除对象时正确定位到派生类的析构函数,并按照正确的顺序销毁对象。 构造函数和析构函数都可以调用虚函数,但需要注意的是,在构造函数中调用虚函数时可能会导致意外的行为,因为在构造函数执行期间,对象的派生类部分尚未初始化。因此,最好在构造函数中避免调用虚函数。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C++虚函数纯虚函数的问题总结](https://blog.csdn.net/weixin_44477424/article/details/124526204)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值