c++函数 和java的区别,虚函数 纯虚函数

11章 虚函数和纯虚函数

一、类型转换又出了问题

学过Java的人反过来学C++,在继承上往往出很多问题。下面就是Java程序员容易出错的例子。


#include <iostream>
#include <string>
using namespace std;

class CBase {
public:
    void show() {
        cout << "BASE" << endl;
    }
};

class CDerive : public CBase {
public:
    void show() {
        cout << "DERIVE" << endl;
    }
};

int main ( )
{
    CDerive d;
    CBase& x = d;

    d.show();
    x.show();

    return 0;
}

变量x是变量d的别名,应该有相同的输出结果,问题出在哪儿呢?问题出在22行赋值的时候。赋值时进行了类型转换(我感觉描述不是很准确),将CDerive类型转换成了CBase类型。

修改部分:java中的函数都是虚函数,子类可以override它(从上到下看(继承层次),你的是你的我的是我的,从下往上看,我的是我的,你的是我的),而c++函数在不加virtual修饰符的时候,子类会overwrite它(你的是你的我的是我的),关键在于指针所指向的位置。

上图d指向的CDriver,而x指向的是CBase为什么?

因为初始化CDriver过程是 1初始化CBase

       2初始化扩充和指针替换(请允许我这么称谓,所谓扩充是构造基类不包含的指针,这些指针可以指向变量或者函数,替换就是对于非虚函数,要把该函数指针指向自己,实际这个动作时在运行时才进行替换的,运行时多态,运行时发现左值和右值类型不用,指针进行替换)

x在创建的时候,实际就是CDriver执行的初始化的基类,就是它自己的指针,并没有强转(因为在运行时发现左值和右值类型相同不用进行转换)。

二、虚函数(java默认函数方式)

解决上面的问题很简单,基类中的show()函数改成虚函数,在派生类中用同名函数“实装”基类的虚函数。如果派生类中没有同名函数,则仍用基类的虚函数实装。


#include <iostream>
#include <string>
using namespace std;

class CBase {
public:
    virtual void show() {
        cout << "BASE SHOW" << endl;
    }
    virtual void print() {
        cout << "BASE PRINT" << endl;
    }
};

class CDerive : public CBase {
public:
    void show() {
        cout << "DERIVE SHOW" << endl;
    }
};

int main ( )
{
    CDerive d;
    CBase& x = d;

    d.show(),  x.show();    //虚函数只影响派生类的实装
    d.print(), x.print();

    CBase b;
    b.show(),  b.print();   //虚函数不影响基类自身的实例

    return 0;
}

注意:“虚继承”和“虚函数”是二个不同概念,“虚继承”针对基类的成员变量的二义性,而“虚函数”是针对成员函数的二义性。“虚继承”是将“virtual”加在派生类的定义处,“虚函数”是将“virtual”加在基类的成员函数的定义处。

修改处:

此次由继承类赋值给父类时,父类方法被overide了,虽然左右值类型相同,但是你的没了,只能用我的了。

三、纯虚函数(java抽象类,接口)

虚继承的应用使程序设计发生了很大变化,也就是在设计阶段只定义基类,而编程阶段通过继承来完成真正的功能。为了控制基类的成员函数在基类什么也不做,引进了“纯虚函数”的概念,使之没有定义体,因而也就不能产生实例。包含“纯虚函数”的类,因不能产生实例,所以称之为“抽象类”。

Java程序员注意:在Java中有类似“抽象类”的称之为“接口”的类,在Visual C++中也有,但这不是标准C++的内容。

修改:

这个和虚函数没啥区别,只不过要求子类必须实现。基类实现没有意义,这个可以认为和java一致


#include <iostream>
#include <string>
using namespace std;

class CBase {
public:
    virtual void show() = 0;    //这是“纯虚函数”
    virtual void print() {      //这是普通的“虚函数”
        cout << "BASE PRINT" << endl;
    }
};

class CDerive : public CBase {
public:
    void show() {      //在派生类中,对“纯虚函数”必须实装
        cout << "DERIVE SHOW" << endl;
    }
};

int main ( )
{
    CDerive d;
    CBase& x = d;

    d.show(),  x.show();
    d.print(), x.print();

    //CBase b;     //抽象类不能实例化
    //b.show(),  b.show();

    return 0;
}
转载自劝学网,并做了修改。不对的地方请斧正。谢谢!
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值