C++中的 虚函数 纯虚函数 虚基类(virtual)


前言:需要了解三者的区别,必须要掌握多态的三个必要条件:

  1. 继承
  2. 重载
  3. 父类指针指向子类对象。

虚函数 纯虚函数 虚基类三者区别

1.虚函数是用于多态中virtual修饰父类函数,确保父类指针调用子类对象时,运行子类函数的。
2.纯虚函数是用来定义接口的,也就是基类中定义一个纯虚函数,基类不用实现,让子类来实现。
3.虚基类是用来在多继承中,比如菱形继承中,如果两个父类继承自同一个类,就只实例化一个父类

①虚函数
第一个是没有使用多态(只用继承)的一般实现方式:

class A
{
public:
    void printf(){
        cout<<"printf A"<<endl;
    }
};
class B : public A
{
public:
    void printf(){
        cout<<"printf B"<<endl;
    }
};
int main(int argc, const char * argv[])
{
    A *a = new A();
    a->printf();
    B *b = new B();
    b->printf();
    return 0;
}  

结果:

printf A
printf B

这是早期没有多态的代码,缺点:代码冗余
下面是使用了多态但是没有引用virtual关键字的情况:

int main(int argc, const char * argv[])
{
    A *a = new B();
    a->printf();
    return 0;
}  

结果:

printf A

因为类的定义都一样,所以就没再写出来了,当父类指针指向子类对象的时候,如果不使用virtual,父类调用方法的时候还是调用了父类自己的方法,没有调用子类重写的方法,所以就没有实现到多态的作用,我们再来在父类中试试加入virtual关键字看看:

class A
{
public:
virtual void printf(){
        cout<<"printf A"<<endl;
    }
};
class B : public A
{
public:
//子类也可以不使用virtual,直接写为void printf()
    virtual void printf(){
        cout<<"printf B"<<endl;
    }
};
int main(int argc, const char * argv[])
{
    A *a = new B();
    a->printf();
    return 0;
}  

结果

printf B

virtual是加入到父类中的,子类的代码尽量加上virtual(方便被继承的子类实现多态),也可以不加,main函数还是父类指针指向子类对象,结果终于可以打印到子类重写的方法了,所以证实了虚函数是用于多态中virtual修饰父类该重写的函数,确保父类指针调用子类对象时运行子类函数的。
② 纯虚函数
纯虚函数就是抽象接口,使用了纯虚函数的类不能被实例化,定义了纯虚函数的类不用写纯虚函数的实现,由子类实现,下面看代码:

class A
{
public:
    virtual void printf() =0;
};
void A::printf()//纯虚函数可以不写实现
{
    cout<<"printf A"<<endl;
}
class B : public A
{
public:
    void printf(){
        cout<<"printf B"<<endl;
    }
};
int main(int argc, const char * argv[])
{
    A *a =new A();//编译出错,纯虚函数的类不能实例化
    a->printf();
    return 0;
}  

virtual void printf() = 0;这是虚函数的写法,我在下面写了虚函数的实现 void A::printf(),其实写不写都没关系,写了也起不了作用,然后我才main函数中尝试吧纯虚函数的类实例化,结果直接报错,说明纯虚函数是不能实例化的。

int main(int argc, const char * argv[])
{
    A *a =newB();//这里使用了多态
    a->printf();
    return 0;
}  

结果:

printf B

把main函数的a指向了子类的对象,结果可以正确打印出子类的方法。由此说明了纯虚函数也是为多态服务的,它的作用是定义一个接口,让子类去实现。
③虚基类
虚基类是c++独有的东西,因为c++中有多继承,也是关键字virtual相关的定义。

先来说说多继承,如果爷爷类(暂把父类的父类暂定为爷爷类 ),父类继承自爷爷类。如果孙类继承自多个父类(听起来有点怪异),那么如果不使用虚基类,就会实例化多个爷爷类对象(越说越离奇),编译器会报错,说有歧义性。如果父类继承自虚基类,则可以解决多个父类不会实例化多个爷爷的问题,就是只有一个爷爷。

class Grandfather{
public:
    int flag;
    Grandfather(){
        flag = 1;
    }
};
class Father1:public Grandfather{
public:
    Father1(){
        flag = 2;
    }
};
class Father2:public Grandfather{
public:
    Father2(){
        flag = 3;
    }
};
class Son:public Father1,public Father2{
};
int main(int argc, const char * argv[])
{
    Son *son = new Son();
    cout<<son->flag<<endl;//这里编译错误,flag访问不明确,因为两个父类中都有flag变量,歧义
    return 0;
}  

如果没有使用虚基类,多个父类继承自同一个爷爷类,就会产生歧义,为了不产生歧义,代码可改为(治标不治本):

cout<<son->Father1::flag<<endl;
cout<<son->Father2::flag<<endl;

如果父类继承虚基类就不同了:

class Grandfather{
public:
    int flag;
    Grandfather(){
        flag = 1;
        cout<<"Grandfather flag = "<<flag <<endl;
    }
};
class Father1:virtual public Grandfather{
public:
    Father1(){
        flag = 2;
        cout<<"Father1 flag = "<<flag<<endl;
    }
};
class Father2:virtual public Grandfather{
public:
    Father2(){
        flag = 3;
        cout<<"Father2 flag = "<<flag<<endl;
    }
};
class Son:public Father1,public Father2{
};
int main(int argc, const char * argv[])
{
    Son *son = new Son();
    cout<<son->flag<<endl;
    return 0;
}  

结果:

Grandfather flag = 1

Father1 flag = 2

Father2 flag = 3

3

现在,可以运行了,class Father2:virtual public Grandfather,就是继承虚基类的写法,爷爷对象只有一个,爷爷类的变量也只实例化了一次,那为什么最后打印出来的是3呢?看构造函数的顺序就可以看出来了,现在构造了爷爷类,再构造第一个继承的父类,最后继承第二个继承的父类,因此flag最后保持在第二个父类的修改值里了。

总的来说,虚函数 ,纯虚函数是为了多态服务,虚基类是为了只实例化一次基类存在的

原文地址:https://blog.51cto.com/14233078/2451973

时间: 2024-10-08 00:15:21

C++中的 虚函数 纯虚函数 虚基类(virtual)的相关文章

winform中利用反射实现泛型数据访问对象基类

考虑到软件使用在客户端,同时想简化代码的实现,就写了一个泛型的数据访问对象基类,并不是特别健全,按道理应该参数化的方式实现insert和update,暂未使用参数化,抽时间改进. /// <summary> /// DAO基类 实体名必须要与数据表字段名一致 /// </summary> /// <typeparam name="T"></typeparam> public class BaseDao<T> where T :

C++:抽象基类和纯虚函数的理解

转载地址:http://blog.csdn.net/acs713/article/details/7352440 抽象类是一种特殊的类,它是为了抽象和设计的目的为建立的,它处于继承层次结构的较上层. ⑴抽象类的定义: 称带有纯虚函数的类为抽象类. ⑵抽象类的作用: 抽象类的主要作用是将有关的操作作为结果接口组织在一个继承层次结构中,由它来为派生类提供一个公共的根,派生类将具体实现在其基类中作为接口的操作.所以派生类实际上刻画了一组子类的操作接口的通用语义,这些语义也传给子类,子类可以具体实现这些

C++纯虚函数、虚函数、实函数、抽象类,重载、重写、重定义

首先,面向对象程序设计(object-oriented programming)的核心思想是数据抽象.继承.动态绑定.通过数据抽象,可以使类的接口与实现分离,使用继承,可以更容易地定义与其他类相似但不完全相同的新类,使用动态绑定,可以在一定程度上忽略相似类的区别,而以统一的方式使用它们的对象. 虚函数的作用是实现多态性(Polymorphism),多态性是将接口与实现进行分离,采用共同的方法,但因个体差异而采用不同的策略.纯虚函数则是一种特殊的虚函数.虚函数联系到多态,多态联系到继承. 一.虚函

【转载】 C++多继承中重写不同基类中相同原型的虚函数

本篇随笔为转载,原文地址:C++多继承中重写不同基类中相同原型的虚函数. 在C++多继承体系当中,在派生类中可以重写不同基类中的虚函数.下面就是一个例子: class CBaseA { public: virtual void TestA(); }; class CBaseB { public: virtual void TestB(); }; class CDerived : public CBaseA, public CBaseB { public: virtual void TestA()

C++ Primer 学习笔记_34_面向对象编程(5)--虚函数与多态(二):纯虚函数、抽象类、虚析构函数、动态创建对象

C++ Primer 学习笔记_34_面向对象编程(5)--虚函数与多态(二):纯虚函数.抽象类.虚析构函数.动态创建对象 一.纯虚函数 1.虚函数是实现多态性的前提 需要在基类中定义共同的接口 接口要定义为虚函数 2.如果基类的接口没办法实现怎么办? 如形状类Shape 解决方法 将这些接口定义为纯虚函数 3.在基类中不能给出有意义的虚函数定义,这时可以把它声明成纯虚函数,把它的定义留给派生类来做 4.定义纯虚函数: class <类名> { virtual <类型> <函

C++ 虚函数&amp;纯虚函数&amp;抽象类&amp;接口&amp;虚基类(转)

http://www.cnblogs.com/fly1988happy/archive/2012/09/25/2701237.html 1. 多态 在面向对象语言中,接口的多种不同实现方式即为多态.多态是指,用父类的指针指向子类的实例(对象),然后通过父类的指针调用实际子类的成员函数. 多态性就是允许将子类类型的指针赋值给父类类型的指针,多态是通过虚函数实现的. 多态可以让父类的指针有“多种形态”,这是一种泛型技术.(所谓泛型技术,就是试图使用不变的代码来实现可变的算法). 2. 虚函数 2.1

虚函数/纯虚函数/抽象类/接口/虚基类

1.多态 在面向对象语言中,接口的多种不同实现方式即为多态.多态是指,用父类的指针指向子类的实例(对象),然后通过父类的指针调用实际子类的成员函数. 在Java中,没有指针,就直接用父类实例化子类对象 多态性就是允许将子类类型的指针赋值给父类类型的指针,多态是通过虚函数实现的,多态可以让父类的指针有“多种形态”,这是一种泛型技术. 所谓泛型技术,就是试图使用不变的代码来实现可变的算法 2.虚函数 在基类的类定义中,定义虚函数的一般形式: Virtual 函数返回值类型 虚函数名(形参表){ 函数

纯虚函数与抽象基类

这一节我们主要介绍一下抽象基类与纯虚函数的关系 首先,我们来看看纯虚函数 1.纯虚函数的定义   纯虚函数是在基类中声明的虚函数,它在基类中没有定义,但要求任何派生类都要定义自己的实现方法.在基类中实现纯虚函数的方法是在函数原型后加"=0".如下     virtual void funtion()=0 2.定义纯虚函数的目的和意义 定义纯虚函数是为了实现一个接口,起到一个规范的作用.纯虚函数的意义在于:让所有的类对象(主要是派生类对象)都可以执行纯虚函数的动作,但类无法为纯虚函数提供

C++基础知识 基类指针、虚函数、多态性、纯虚函数、虚析构

一.基类指针.派生类指针 父类指针可以new一个子类对象 二.虚函数 有没有一个解决方法,使我们只定义一个对象指针,就可以调用父类,以及各个子类的同名函数? 有解决方案,这个对象指针必须是一个父类类型,我们如果想通过一个父类指针调用父类.子类中的同名函数的话,这个函数是有要求的: 在父类中,eat函数声明之前必须要加virtual声明eat()函数为虚函数. 一旦某个函数被声明为虚函数,那么所有派生类(子类)中eat()函数都是虚函数. 为了避免你在子类中写错虚函数,在C++11中,你可以在函数

C++反汇编第二讲,反汇编中识别虚表指针,以及指向的虚函数地址

讲解之前,了解下什么是虚函数,什么是虚表指针,了解下语法,(也算复习了) 开发知识为了不码字了,找了一篇介绍比较好的,这里我扣过来了,当然也可以看原博客链接: http://blog.csdn.net/hackbuteer1/article/details/7558868 一丶虚函数讲解(复习开发,熟悉内存模型) 1.复习开发知识 首先:强调一个概念 定义一个函数为虚函数,不代表函数为不被实现的函数. 定义他为虚函数是为了允许用基类的指针来调用子类的这个函数. 定义一个函数为纯虚函数,才代表函数