发现Java和C++多态性的一个区别

王朝c/c++·作者佚名  2006-01-09
窄屏简体版  字體: |||超大  

先来看一下这两个程序:

代码1:

//TestA.java

class A {

protected int num;

public A() {

init();

num = 1;

}

protected void init(){

System.out.println("i am A");

System.out.println(num);

}

}

class B extends A {

public B(){

init();

num = 2;

}

protected void init(){

System.out.println("i am B");

System.out.println(num);

}

}

class TestA {

public static void main(String[] args) {

B b = new B();

}

}

------

代码2:

//TestA.cpp

#include <iostream>

using namespace std;

class A {

protected:

int num;

public:

A() {

init();

num = 1;

}

protected:

virtual void init() {

cout << "i am A" << endl;

cout << num << endl;

}

};

class B:public A {

public:

B() {

init();

num = 2;

}

protected:

void init() {

cout << "i am B" << endl;

cout << num << endl;

}

};

int main() {

B b;

return 0;

}

------

代码1的编译运行结果如下(j2sdk 1.4.2):

i am B

0

i am B

1

代码2的编译运行结果如下(g++ 3.3.1):

i am A

0

i am B

1

发现不同了吧。

对于代码1,java采用的是动态绑定技术。构建子类B对象的时候,先调用基类A的构造函数,而A的构造函数中调用了init()方法,此时对java来说还是在构建B对象的过程中,因此这个调用的init()方法不是基类A中的而是B中的。

对于代码2,有点不同。虚函数在C++中原本是用来实现动态联编的,但在构造函数中调用虚函数时,采用的不是动态联编而是静态联编,即基类A的构造函数中的init()调用的是自己的init()方法。

另:把TestA.cpp中的virtual关键字去掉,看看运行结果,和没有去掉之前不一样,这是什么原因?有谁知道能告诉我吗?

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航