纯虚函数和虚函数

来源:互联网 发布:兰州大学网络登录入口 编辑:程序博客网 时间:2024/06/04 17:48
如下声明表示一个函数为纯虚函数:
class A
{
public:
    virtual void foo()=0;   // =0标志一个虚函数为纯虚函数
};

    一个函数声明为纯虚后,纯虚函数的意思是:我是一个抽象类!不要把我实例化!纯虚函数用来规范派生类的行为,实际上就是所谓的“接口”。它告诉使用者,我的派生类都会有这个函数。

* 有纯虚函数的类是抽象类,不能生成对象,只能派生。他派生的类的纯虚函数没有被改写,那么,它的派生类还是个抽象类。

*定义纯虚函数就是为了让基类不可实例化化,
    因为实例化这样的抽象数据结构本身并没有意义.
    或者给出实现也没有意义
    实际上纯虚函数的引入,可能是出于两个目的,
    1.为了安全.因为避免任何需要明确但是因为不小心而导致的未知的结果. 提醒子类去做应做的实现.
    2.为了效率,不是程序执行的效率,而是为了编码的效率.

* 当一个类打算被用作其它类的基类时,它的析构函数必须是虚的

那么,什么是虚函数呢,我们先来看看微软的解释:

虚函数是指一个类中你希望重载的成员函数,当你用一个基类指针或引用指向一个继承类对象的时候,你调用一个虚函数,实际调用的是继承类的版本。

——摘自MSDN

例子:
#i nclude "stdio.h"
#i nclude "conio.h"
class Parent
{
public:
    char data[20];
    void Function1();
    virtual void Function2(); // 这里声明Function2是虚函数
}parent;

void Parent::Function1()
{
    printf("This is parent,function1\n");
}

void Parent::Function2()
{
    printf("This is parent,function2\n");
}

class Child:public Parent
{
    void Function1();
    void Function2();
} child;

void Child::Function1()
{
    printf("This is child,function1\n");
}

void Child::Function2()
{
    printf("This is child,function2\n");
}

int main(int argc, char* argv[])
{
    Parent *p; // 定义一个基类指针
    if(_getch()=='c') // 如果输入一个小写字母c
        p=&child; // 指向继承类对象
    else
        p=&parent; // 否则指向基类对象
    
    p->Function1(); // 这里在编译时会直接给出Parent::Function1()的入口地址。
    p->Function2(); // 注意这里,执行的是哪一个Function2?
    return 0;
}

用任意版本的Visual C++或Borland C++编译并运行,输入一个小写字母c,得到下面的结果:
This is parent,function1
This is child,function2
为什么会有第一行的结果呢?因为我们是用一个Parent类的指针调用函数Fuction1(),虽然实际上这个指针指向的是Child类的对象,但编译器无法知道这一事实(直到运行的时候,程序才可以根据用户的输入判断出指针指向的对象),它只能按照调用Parent类的函数来理解并编译,所以我们看到了第一行的结果。

那么第二行的结果又是怎么回事呢?我们注意到,Function2()函数在基类中被virtual关键字修饰,也就是说,它是一个虚函数。虚函数最关键的特点是“动态联编”,它可以在运行时判断指针指向的对象,并自动调用相应的函数。

如果我们在运行上面的程序时任意输入一个非c的字符,结果如下:
This is parent,function1
This is parent,function2
0 0