`
xm_king
  • 浏览: 395337 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
博客专栏
Group-logo
Spring技术内幕读书笔...
浏览量:15658
社区版块
存档分类
最新评论

Java动态绑定机制的内幕(转载)

    博客分类:
  • JAVA
阅读更多

在Java方法调用的过程中,JVM是如何知道调用的是哪个类的方法源代码? 这里面到底有什么内幕呢? 这篇文章我们就将揭露JVM方法调用的静态(static binding) 动态绑定机制(auto binding)

 

 

★ 静态绑定机制

Java代码  收藏代码
  1. //被调用的类  
  2. package hr.test;  
  3. class Father{  
  4.       public static void f1(){  
  5.               System.out.println("Father— f1()");  
  6.       }  
  7. }  
  8. //调用静态方法  
  9. import hr.test.Father;  
  10. public class StaticCall{  
  11.        public static void main(){  
  12.             Father.f1(); //调用静态方法  
  13.        }  
  14. }  

     上面的源代码中执行方法调用的语句(Father.f1())被编译器编译成了一条指令:invokestatic #13。我们看看JVM是如何处理这条指令的

     (1) 指令中的#13指的是StaticCall类的常量池中第13个常量表的索引项(关于常量池详见《Class文件内容及常量池 》)。这个常量表(CONSTATN_Methodref_info ) 记录的是方法f1信息的符号引用(包括f1所在的类名,方法名和返回类型)。JVM会首先根据这个符号引用找到方法f1所在的类的全限定名: hr.test.Father。

     (2) 紧接着JVM会加载、链接和初始化Father类。

     (3) 然后在Father类所在的方法区中找到f1()方法的直接地址,并将这个直接地址记录到StaticCall类的常量池索引为13的常量表中。这个过程叫常量池解析 ,以后再次调用Father.f1()时,将直接找到f1方法的字节码。

     (4) 完成了StaticCall类常量池索引项13的常量表的解析之后,JVM就可以调用f1()方法,并开始解释执行f1()方法中的指令了。

 

     通过上面的过程,我们发现经过常量池解析之后,JVM就能够确定要调用的f1()方法具体在内存的什么位置上了。实际上,这个信息在编译阶段就已经在StaticCall类的常量池中记录了下来。这种在编译阶段就能够确定调用哪个方法的方式,我们叫做 静态绑定机制

 

     除了被static 修饰的静态方法,所有被private 修饰的私有方法、被final 修饰的禁止子类覆盖的方法都会被编译成invokestatic指令。另外所有类的初始化方法<init>和<clinit>会被编译成invokespecial指令。JVM会采用静态绑定机制来顺利的调用这些方法。

 

 

 

★ 动态绑定机制

Java代码  收藏代码
  1. package hr.test;  
  2. //被调用的父类  
  3. class Father{  
  4.     public void f1(){  
  5.         System.out.println("father-f1()");  
  6.     }  
  7.         public void f1(int i){  
  8.                 System.out.println("father-f1()  para-int "+i);  
  9.         }  
  10. }  
  11. //被调用的子类  
  12. class Son extends Father{  
  13.     public void f1(){ //覆盖父类的方法  
  14.         System.out.println("Son-f1()");  
  15.     }  
  16.         public void f1(char c){  
  17.                 System.out.println("Son-s1() para-char "+c);  
  18.         }  
  19. }  
  20.   
  21. //调用方法  
  22. import hr.test.*;  
  23. public class AutoCall{  
  24.     public static void main(String[] args){  
  25.         Father father=new Son(); //多态  
  26.         father.f1(); //打印结果: Son-f1()  
  27.     }  
  28. }  

      上面的源代码中有三个重要的概念:多态(polymorphism) 方法覆盖方法重载 。打印的结果大家也都比较清楚,但是JVM是如何知道f.f1()调用的是子类Sun中方法而不是Father中的方法呢?在解释这个问题之前,我们首先简单的讲下JVM管理的一个非常重要的数据结构——方法表

 

       在JVM加载类的同时,会在方法区中为这个类存放很多信息(详见《Java 虚拟机体系结构 》)。其中就有一个数据结构叫方法表。它以数组的形式记录了当前类及其所有超类的可见方法字节码在内存中的直接地址 。下图是上面源代码中Father和Sun类在方法区中的方法表:

 

      上图中的方法表有两个特点:(1) 子类方法表中继承了父类的方法,比如Father extends Object。 (2) 相同的方法(相同的方法签名:方法名和参数列表)在所有类的方法表中的索引相同。比如Father方法表中的f1()和Son方法表中的f1()都位于各自方法表的第11项中。

 

      对于上面的源代码,编译器首先会把main方法编译成下面的字节码指令:

多态调用的字节码指令代码  收藏代码
  1. 0  new hr.test.Son [13] //在堆中开辟一个Son对象的内存空间,并将对象引用压入操作数栈  
  2. 3  dup    
  3. 4  invokespecial #7 [15] // 调用初始化方法来初始化堆中的Son对象   
  4. 7  astore_1 //弹出操作数栈的Son对象引用压入局部变量1中  
  5. 8  aload_1 //取出局部变量1中的对象引用压入操作数栈  
  6. 9  invokevirtual #15 //调用f1()方法  
  7. 12  return  

       其中invokevirtual指令的详细调用过程是这样的:

       (1) invokevirtual指令中的#15指的是AutoCall类的常量池中第15个常量表的索引项。这个常量表(CONSTATN_Methodref_info ) 记录的是方法f1信息的符号引用(包括f1所在的类名,方法名和返回类型)。JVM会首先根据这个符号引用找到调用方法f1的类的全限定名: hr.test.Father。这是因为调用方法f1的类的对象father声明为Father类型。

       (2) 在Father类型的方法表中查找方法f1,如果找到,则将方法f1在方法表中的索引项11(如上图)记录到AutoCall类的常量池中第15个常量表中(常量池解析 )。这里有一点要注意:如果Father类型方法表中没有方法f1,那么即使Son类型中方法表有,编译的时候也通过不了。因为调用方法f1的类的对象father的声明为Father类型。

       (3) 在调用invokevirtual指令前有一个aload_1指令,它会将开始创建在堆中的Son对象的引用压入操作数栈。然后 invokevirtual指令会根据这个Son对象的引用首先找到堆中的Son对象,然后进一步找到Son对象所属类型的方法表。过程如下图所示:

                   

      (4) 这是通过第(2)步中解析完成的#15常量表中的方法表的索引项11,可以定位到Son类型方法表中的方法f1(),然后通过直接地址找到该方法字节码所在的内存空间。

 

      很明显,根据对象(father)的声明类型(Father)还不能够确定调用方法f1的位置,必须根据father在堆中实际创建的对象类型Son来确定f1方法所在的位置。这种在程序运行过程中,通过动态创建的对象的方法表来定位方法的方式,我们叫做 动态绑定机制

 

      上面的过程很清楚的反映出在方法覆盖的多态调用的情况下,JVM是如何定位到准确的方法的。但是下面的调用方法JVM是如何定位的呢?(仍然使用上面代码中的Father和Son类型)

Java代码  收藏代码
  1. public class AutoCall{  
  2.        public static void main(String[] args){  
  3.              Father father=new Son();  
  4.              char c='a';  
  5.              father.f1(c); //打印结果:father-f1()  para-int 97  
  6.        }  
  7. }  

       问题是Fahter类型中并没有方法签名为f1(char)的方法呀。但打印结果显示JVM调用了Father类型中的f1(int)方法,并没有调用到Son类型中的f1(char)方法。

 

       根据上面详细阐述的调用过程,首先可以明确的是:JVM首先是根据对象father声明的类型Father来解析常量池的(也就是用Father方法表中的索引项来代替常量池中的符号引用)。如果Father中没有匹配到"合适" 的方法,就无法进行常量池解析,这在编译阶段就通过不了。

      那么什么叫"合适"的方法呢?当然,方法签名完全一样的方法自然是合适的。但是如果方法中的参数类型在声明的类型中并不能找到呢?比如上面的代码中调用father.f1(char),Father类型并没有f1(char)的方法签名。实际上,JVM会找到一种“凑合”的办法,就是通过 参数的自动转型 来找 到“合适”的 方法。比如char可以通过自动转型成int,那么Father类中就可以匹配到这个方法了 (关于Java的自动转型问题可以参见《【解惑】Java类型间的转型 》)。但是还有一个问题,如果通过自动转型发现可以“凑合”出两个方法的话怎么办?比如下面的代码:

Java代码  收藏代码
  1. class Father{  
  2.     public void f1(Object o){  
  3.         System.out.println("Object");  
  4.     }  
  5.     public void f1(double[] d){  
  6.         System.out.println("double[]");  
  7.     }  
  8.       
  9. }  
  10. public class Demo{  
  11.     public static void main(String[] args) {  
  12.         new Father().f1(null); //打印结果: double[]  
  13.     }  
  14. }  

        null可以引用于任何的引用类型,那么JVM如何确定“合适”的方法呢。一个很重要的标准就是:如果一个方法可以接受传递给另一个方法的任何参数,那么第一个方法就相对不合适。比如上面的代码: 任何传递给f1(double[])方法的参数都可以传递给f1(Object)方法,而反之却不行,那么f1(double[])方法就更合适。因此JVM就会调用这个更合适的方法。

 

 

★ 总结

 

      (1) 所有私有方法、静态方法、构造器及初始化方法<clinit>都是采用静态绑定机制。在编译器阶段就已经指明了调用方法在常量池中的符号引用,JVM运行的时候只需要进行一次常量池解析即可。


      (2) 类对象方法的调用必须在运行过程中采用动态绑定机制。

           首先,根据对象的声明类型(对象引用的类型)找到“合适”的方法。具体步骤如下:

           ① 如果能在声明类型中匹配到方法签名完全一样(参数类型一致)的方法,那么这个方法是最合适的。

           ② 在第①条不能满足的情况下,寻找可以“凑合”的方法。标准就是通过将参数类型进行自动转型之后再进行匹配。如果匹配到多个自动转型后的方法签名f(A)和 f(B),则用下面的标准来确定合适的方法:传递给f(A)方法的参数都可以传递给f(B),则f(A)最合适。反之f(B)最合适

 

           ③ 如果仍然在声明类型中找不到“合适”的方法,则编译阶段就无法通过。

 

           然后,根据在堆中创建对象的实际类型找到对应的方法表,从中确定具体的方法在内存中的位置。

 

 

 

★ 覆写(override)

       一个实例方法可以覆写(override)在其超类中可访问到的具有相同签名的所有实例方法,从而使能了动态分派(dynamic dispatch);换句话说,VM将基于实例的运行期类型来选择要调用的覆写方法。覆写是面向对象编程技术的基础,并且是唯一没有被普遍劝阻的名字重用形式:

Java代码  收藏代码
  1. class Base{  
  2.       public void f(){}  
  3. }  
  4. class Derived extends Base{  
  5.       public void f(){}  
  6. }  

 

 

★ 隐藏(hide)

       一个域、静态方法或成员类型可以分别隐藏(hide)在其超类中可访问到的具有相同名字(对方法而言就是相同的方法签名)的所有域、静态方法或成员类型。隐藏一个成员将阻止其被继承。

Java代码  收藏代码
  1. class Base{  
  2.       public static void f(){}  
  3. }  
  4. class Derived extends Base  {  
  5.       private static void f(){}   //hides Base. f()  
  6. }  

 

★ 重载(overload)
      在某个类中的方法可以重载(overload)另一个方法,只要它们具有相同的名字和不同的签名。由调用所指定的重载方法是在编译期选定的。

Java代码  收藏代码
  1. class CircuitBreaker{  
  2.       public void f (int i){}    //int overloading  
  3.       public void f(String s){}   //String overloading  
  4. }  

 

★ 遮蔽(shadow)
     一个变量、方法或类型可以分别遮蔽(shadow)在一个闭合的文本范围内的具有相同名字的所有变量、方法或类型。如果一个实体被遮蔽了,那么你用它的简单名是无法引用到它的;根据实体的不同,有时你根本就无法引用到它。

Java代码  收藏代码
  1. class WhoKnows{  
  2.     static String sentence=”I don't know.”;  
  3.     public static void main(String[] args〕{  
  4.            String sentence=”I don't know.”;  //shadows static field  
  5.            System.out. println (sentence);  // prints local variable  
  6.     }  
  7. }  

      尽管遮蔽通常是被劝阻的,但是有一种通用的惯用法确实涉及遮蔽。构造器经常将来自其所在类的某个域名重用为一个参数,以传递这个命名域的值。这种惯用法并不是没有风险,但是大多数Java程序员都认为这种风格带来的实惠要超过
其风险:

Java代码  收藏代码
  1. class Belt{  
  2.       private find int size ;  //Parameter shadows Belt. size  
  3.       public Belt (int size){  
  4.            this. size=size;  
  5.       }  
  6. }  

 

★ 遮掩(obscure)

       一个变量可以遮掩具有相同名字的一个类型,只要它们都在同一个范围内:如果这个名字被用于变量与类型都被许可的范围,那么它将引用到变量上。相似地,一个变量或一个类型可以遮掩一个包。遮掩是唯一一种两个名字位于不同的名字空间的名字重用形式,这些名字空间包括:变量、包、方法或类型。如果一个类型或一个包被遮掩了,那么你不能通过其简单
名引用到它,除非是在这样一个上下文环境中,即语法只允许在其名字空间中出现一种名字。遵守命名习惯就可以极大地消除产生遮掩的可能性:

Java代码  收藏代码
  1. public class Obscure{  
  2.       static String System;// Obscures type java.lang.System  
  3.       public static void main(String[] args)  
  4.             // Next line won't compile:System refers to static field  
  5.             System. out. println(“hello, obscure world!”);  
  6.       }  
  7. }  
分享到:
评论
3 楼 qiemengdao 2012-02-17  
很精彩,学习了。
2 楼 zuchunlei2010 2011-09-15  
Very Good.很精彩!
1 楼 waterenjoy 2011-06-23  
精彩!学习了,谢谢!

相关推荐

    Java动态绑定机制的内幕

    这篇文章我们将揭露JVM方法调用的静态(static binding) 和动态绑定机制(auto binding) 。  静态绑定机制 //被调用的类 package hr.test; class Father{ public static void f1(){ System.out.println(...

    Java动态绑定和静态绑定.doc

    在Java编程语言中,动态绑定和静态绑定是两个重要的概念,它们涉及到方法调用和多态性。了解这两个概念对于深入理解面向对象编程至关重要。本文将详细探讨它们的区别,并通过实例来阐述它们的工作原理。 首先,让...

    Java动态绑定和内联

    Java动态绑定和内联是Java编程中的两个关键概念,它们对于理解Java的面向对象特性,尤其是多态性和性能优化有着重要的作用。 动态绑定,又称晚期绑定或运行时绑定,是Java语言的一个核心特性。它指的是在运行时而非...

    java动态绑定和静态绑定用法实例详解

    Java中存在两种类型的绑定机制:静态绑定和动态绑定。 静态绑定是指在程序执行以前已经被绑定(即在编译过程中就已经知道这个方法到底是哪个类中的方法)。在Java中,final、static、private修饰的方法和构造方法是...

    Java中的静态绑定和动态绑定Java开发Java经验技巧

    在Java编程语言中,静态绑定和动态绑定是两个至关重要的概念,它们关乎程序的执行效率和灵活性。了解这两个概念对于提升Java开发技能至关重要。 首先,我们来解释一下静态绑定(也称为早期绑定)。静态绑定主要涉及...

    java代码绑定时间

    综上所述,合理利用Oracle的绑定变量机制是Java开发者优化数据库性能的重要手段之一。它不仅可以避免不必要的硬解析,减轻CPU和闩锁资源的压力,还能优化共享池的管理,提升系统的并发处理能力和响应速度。在实际...

    Java动态绑定的方法重载的实现.zip

    总之,Java中的动态绑定和方法重载是实现面向对象编程的重要机制,它们共同为开发者提供了更强大、更灵活的代码设计工具。理解并熟练运用这两个概念,将有助于编写出更加高效、可维护的Java代码。

    Java动态绑定的方法重载的实现.pdf

    Java动态绑定的方法重载的实现.pdf

    理解Java中的静态绑定和动态绑定

    在Java编程语言中,静态绑定和动态绑定是两种不同的方法调用机制,它们与面向对象编程密切相关。这两种绑定方式决定了程序在运行时如何选择和执行特定的方法。 静态绑定(早期绑定),也称为编译时绑定,是指在程序...

    Java 静态绑定与动态绑定深入分析

    Java中的静态绑定与动态绑定是面向对象编程中的关键概念,它们决定了方法调用的方式和时机。静态绑定(也称为早期绑定或编译时绑定)和动态绑定(也称为晚期绑定或运行时绑定)是Java中多态性实现的基石。 **静态...

    xml和java绑定

    XML(可扩展标记语言)与Java的绑定是指在Java应用程序中解析、操作和生成XML文档的过程。这种绑定允许Java对象和XML数据之间直接映射,简化了数据交换和处理。以下是一些关于XML和Java绑定的关键知识点: 1. **...

    java的映射机制

    ### Java的映射机制深入解析 Java的映射机制是其框架设计与应用开发中的核心概念之一,尤其...无论是数据绑定、异常处理还是业务流程控制,映射机制都发挥着不可或缺的作用,是每个Java开发者应该掌握的关键技术之一。

    Java绑定XML架构的技术内涵

    ### Java绑定XML架构的技术内涵 #### 一、引言 在现代软件开发中,XML(Extensible Markup Language)被广泛应用于数据交换与存储。而Java作为一种主流的编程语言,在处理XML方面有着丰富的工具和技术支持。Java...

    Delphi 动态绑定事件处理函数过程

    在 Delphi 编程环境中,动态绑定事件处理函数是一个强大的特性,它允许程序在运行时动态地连接事件和对应的处理函数,而不是在编译时静态地确定。这为开发提供了更大的灵活性,特别是在处理不确定数量的对象或者需要...

    在Java中实现Oracle变量的绑定方法 .doc

    在Java中实现Oracle变量的绑定方法,主要是为了优化SQL语句的执行效率,减少数据库解析负担,提升系统性能。Oracle数据库在处理SQL语句时,会先进行解析,然后将解析后的SQL语句存储在共享池(Shared Pool)中。如果...

    java公众号绑定域名,微信消息接收和token验证

    java实现公众号绑定域名,微信消息接收和token验证功能

    《剑指offer》Java理解静态绑定与动态绑定.pdf

    Java中的静态绑定与动态绑定是面向对象编程中的两个核心概念,它们关系到程序的编译与运行时行为。理解这两个概念对于写出高效、灵活的代码至关重要。 首先,静态绑定(也称为前期绑定或编译时绑定)是指在编译期间...

    Java与XML数据绑定

    Java与XML数据绑定是将XML文档中的数据结构映射到Java对象的过程,反之亦然。这一技术使得在Java应用程序中处理XML数据变得更加便捷,避免了手动解析和构建XML字符串的繁琐工作。本篇将深入探讨Java中XML数据绑定的...

Global site tag (gtag.js) - Google Analytics