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

类别:编程语言 点击:0 评论:0 推荐:


先来看一下这两个程序:

代码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关键字去掉,看看运行结果,和没有去掉之前不一样,这是什么原因?有谁知道能告诉我吗?

本文地址:http://com.8s8s.com/it/it23658.htm