今天看jcvm的标准的 时候,看到有一个virtual method,感觉很疑惑,以前看Java的时候并没有发现有这类方法。

百度、Google了一下,才发现,Java中普通方法就是virtual method,动态绑定是Java的默认行为

如果不想让一个方法成为virtual method,只要把这个方法申明为final就可以了。

 

至于在c++中虚方法是怎么回事,可以参考下面这篇文章

 

在很多的博客中也有这样一种说法,Java的virtual method其实就是dynamic binding/dynamic dispatch.

(http://stackoverflow.com/questions/9453701/what-is-virtual-method-calling-in-java)

从这里面的回答可以看出,virtual method就是为了多态而出现的

 

下面简单的介绍一下动态方法调度。

在Java中,如果一个子类的方法和超类的某个方法具有相同的名称和类型签名,那么称子类中的这个方法重写了超类中相应的方法

当子类中调用被重写的方法时,总是调用由子类定义的版本,由超类定义的版本会被隐藏。

方法重写形成了动态方法调度(dynamic method dispatch)的基础,动态方法调度可以说是Java中最强大的功能

动态方法调度室一种机制,通过这种机制可以在运行时,而不是在编译时解析对重写方法的调用。

动态方法调度就是Java实现多态的机理所在

 

首先申明一点:超类引用变量可以指向子类对象。Java利用这一事实,在运行时解析对重写方法的调用。

下面是实现原理:

当通过超类引用调用重写的方法时,Java根据在调用时所引用对象的类型来判断调用哪个版本的方法。因此,这个决定是在运行时做出的。

如果引用不同类型的对象,就会调用不同版本的方法。

也就是说,是当前正在引用的对象的类型(而不是引用变量的类型)决定了将要执行哪个版本的重写方法。


例子如下:

public class Test {

	  public static void main(String[] args){

		    superTest a = new superTest();
		    a.eat();

		    superTest b = new superTest1();
		    b.eat();

		    superTest c = new superTest2();
		    c.eat();

	  }}
class superTest{

	  void eat(){
		    System.out.println("this is the superTest print");
	  }
}class superTest1 extends superTest{
	  void eat(){
		    System.out.println("this is the superTest1 print ");
	  }
}class superTest2 extends superTest{
	  void eat(){
		    System.out.println("this is the superTest2 print");
	  }
}

最后的输出结果是:

this is the superTest print
this is the superTest1 print
this is the superTest2 print