虚函数的一种重要的应用是在子类重写父类方法上,一般都是在父类中声明的时候用关键字virtual修饰。

在C++中,一个父类的对象指针是可以指向子类的实例化对象,这个时候可以用该对象指针来访问父类的成员函数,但是访问不了子类的成员函数。如果这个父类的对象指针想要访问子类中的函数,则需要在父类中对应的函数添加virtual关键词将该函数变为虚函数,这个时候父类的对象指针就可以访问子类重写父类虚函数:

#include <iostream>

class A
{
public:
    virtual void prt1()
    {
        std::cout << "A1" << std::endl;
    }
    void prt2()
    {
        std::cout << "A2" << std::endl;
    }
protected:
private:
};

class B :public A
{
public:
    void prt1()
    {
        std::cout << "B1" << std::endl;
    }
    void prt2()
    {
        std::cout << "B2" << std::endl;
    }
};


int main(int argc, char* argv[])
{
    A *ptrA = NULL;
    B *ptrB = new B;
    ptrA = ptrB;
    ptrA->prt1();
    ptrA->prt2();
    // delete ptrB; // 这里delete ptrB或者ptrA其中一个就可以了。
    delete ptrA;
    system("pause");
    return 0;
}

代码运行的结果显示如下:

父类A声明了两个函数,prt1是虚函数,prt2是普通的成员函数,子类B继承了父类A,同样有两个函数,prt1是重写了父类的虚函数,prt2是隐藏了父类的普通话成员函数,所以当声明一个父类的对象指针并指向子类的实例化对象的时候,ptrA可以访问B的函数prt1,而访问prt2的时候只能访问父类的prt2函数。

 

提到虚函数就要提一下纯虚函数,纯虚函数跟虚函数类似,也用virtual来修饰,但是多了一个是在函数最后要添加"=0"来表示纯虚函数,比如这样:

class A
{
public:
	virtual void prt0() = 0;
	virtual void prt1()
	{
	    std::cout << "A1" << std::endl;
	}
	void prt2()
	{
	    std::cout << "A2" << std::endl;
	}
protected:
private:
};

在上一个例子的类A中添加一个纯虚函数ptr0,这个时候如果类B中没有对ptr0进行重写就实例化一个对象的话,编译的时候会报错,报错类B不能进行实例化。所以要在B类中重写A的纯虚函数后B才能被实例化:

class B :public A
{
public:
	void prt0()
	{
	    std::cout << "B0" << std::endl;
	}
	void prt1()
	{
	    std::cout << "B1" << std::endl;
	}
	void prt2()
	{
	    std::cout << "B2" << std::endl;
	}
};

这样父类的对象指针指向子类的实例化对象的时候,也还可以访问子类重写的纯虚函数的:

int main(int argc, char* argv[])
{
	A *ptrA = NULL;
	B *ptrB = new B;
	ptrA = ptrB;
	ptrA->prt0();
	ptrA->prt1();
	ptrA->prt2();
	//delete ptrB;
	delete ptrA;
	system("pause");
	return 0;
}

结果显示如下:

所以纯虚函数的总结如下:

 

  • 父类声明纯虚函数的时候在函数最后添加"=0",这个时候父类变成抽象类,抽象类不能被实例化;

  • 子类继承重写抽象类的纯虚函数后子类才可以被实例化,并且在声明的时候不需要virtual修饰,但要与被重写的纯虚函数的函数名、参数列表、返回值完全一致,如果返回值则需要是协变的情况,否则会提示重写虚函数返回类型有差异,且不是来自“A::prt0”的协变

  • 抽象类可以有声明为一个指针指向子类的实例化对象,这个时候可以访问被子类重写的虚函数和纯虚函数,但是普通函数不行。

  • 纯虚函数的作用可以说是“只提供申明,没有实现”,是约束子类的接口的方法。

 

 

月是情人和鬼的魂魄,

月色冰冰燃一盏青焰的长明灯

中元夜,鬼也醒着,人也醒着

人在桥上怔怔地出神