学过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执行的初始化的基类,就是它自己的指针,并没有强转(因为在运行时发现左值和右值类型相同不用进行转换)。
解决上面的问题很简单,基类中的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中有类似“抽象类”的称之为“接口”的类,在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; }
转载自劝学网,并做了修改。不对的地方请斧正。谢谢!