从Java静态绑定和动态绑定中得到优化启示

开发 后端
一个Java程序的执行要经过编译和执行(解释)这两个步骤,同时Java又是面向对象的编程语言。当子类和父类存在同一个方法,子类重写了父类的方法,程序在运行时调用方法是调用父类的方法还是子类的重写方法呢,这应该是我们在初学Java时遇到的问题。这里首先我们将确定这种调用何种方法实现或者变量的操作叫做绑定。

一个Java程序的执行要经过编译和执行(解释)这两个步骤,同时Java又是面向对象的编程语言。当子类和父类存在同一个方法,子类重写了父类的方法,程序在运行时调用方法是调用父类的方法还是子类的重写方法呢,这应该是我们在初学Java时遇到的问题。这里首先我们将确定这种调用何种方法实现或者变量的操作叫做绑定。

在Java中存在两种绑定方式,一种为静态绑定,又称作早期绑定。另一种就是动态绑定,亦称为后期绑定。

区别对比

  • 静态绑定发生在编译时期,动态绑定发生在运行时
  • 使用private或static或final修饰的变量或者方法,使用静态绑定。而虚方法(可以被子类重写的方法)则会根据运行时的对象进行动态绑定。
  • 静态绑定使用类信息来完成,而动态绑定则需要使用对象信息来完成。
  • 重载(Overload)的方法使用静态绑定完成,而重写(Override)的方法则使用动态绑定完成。

重载方法的示例

这里展示一个重载方法的示例。

  1. public class TestMain { 
  2.   public static void main(String[] args) { 
  3.       String str = new String(); 
  4.       Caller caller = new Caller(); 
  5.       caller.call(str); 
  6.   } 
  7.  
  8.   static class Caller { 
  9.       public void call(Object obj) { 
  10.           System.out.println("an Object instance in Caller"); 
  11.       } 
  12.        
  13.       public void call(String str) { 
  14.           System.out.println("a String instance in in Caller"); 
  15.       } 
  16.   } 

执行的结果为

  1. 22:19 $ java TestMain 
  2. a String instance in in Caller 

在上面的代码中,call方法存在两个重载的实现,一个是接收Object类型的对象作为参数,另一个则是接收String类型的对象作为参数。str是一个String对象,所有接收String类型参数的call方法会被调用。而这里的绑定就是在编译时期根据参数类型进行的静态绑定。

验证

光看表象无法证明是进行了静态绑定,使用javap发编译一下即可验证。

  1. 22:19 $ javap -c TestMain 
  2. Compiled from "TestMain.java" 
  3. public class TestMain { 
  4.   public TestMain(); 
  5.     Code: 
  6.        0: aload_0 
  7.        1: invokespecial #1                  // Method java/lang/Object."<init>":()V 
  8.        4return 
  9.  
  10.   public static void main(java.lang.String[]); 
  11.     Code: 
  12.        0new           #2                  // class java/lang/String 
  13.        3: dup 
  14.        4: invokespecial #3                  // Method java/lang/String."<init>":()V 
  15.        7: astore_1 
  16.        8new           #4                  // class TestMain$Caller 
  17.       11: dup 
  18.       12: invokespecial #5                  // Method TestMain$Caller."<init>":()V 
  19.       15: astore_2 
  20.       16: aload_2 
  21.       17: aload_1 
  22.       18: invokevirtual #6                  // Method TestMain$Caller.call:(Ljava/lang/String;)V 
  23.       21return 

看到了这一行18: invokevirtual #6 // Method TestMain$Caller.call:(Ljava/lang/String;)V确实是发生了静态绑定,确定了调用了接收String对象作为参数的caller方法。

重写方法的示例

  1. public class TestMain { 
  2.   public static void main(String[] args) { 
  3.       String str = new String(); 
  4.       Caller caller = new SubCaller(); 
  5.       caller.call(str); 
  6.   } 
  7.    
  8.   static class Caller { 
  9.       public void call(String str) { 
  10.           System.out.println("a String instance in Caller"); 
  11.       } 
  12.   } 
  13.    
  14.   static class SubCaller extends Caller { 
  15.       @Override 
  16.       public void call(String str) { 
  17.           System.out.println("a String instance in SubCaller"); 
  18.       } 
  19.   } 

执行的结果为

  1. 22:27 $ java TestMain 
  2. a String instance in SubCaller 

上面的代码,Caller中有一个call方法的实现,SubCaller继承Caller,并且重写了call方法的实现。我们声明了一个Caller类型的变量callerSub,但是这个变量指向的时一个SubCaller的对象。根据结果可以看出,其调用了SubCaller的call方法实现,而非Caller的call方法。这一结果的产生的原因是因为在运行时发生了动态绑定,在绑定过程中需要确定调用哪个版本的call方法实现。

验证

使用javap不能直接验证动态绑定,然后如果证明没有进行静态绑定,那么就说明进行了动态绑定。

  1. 22:27 $ javap -c TestMain 
  2. Compiled from "TestMain.java" 
  3. public class TestMain { 
  4.   public TestMain(); 
  5.     Code: 
  6.        0: aload_0 
  7.        1: invokespecial #1                  // Method java/lang/Object."<init>":()V 
  8.        4return 
  9.  
  10.   public static void main(java.lang.String[]); 
  11.     Code: 
  12.        0new           #2                  // class java/lang/String 
  13.        3: dup 
  14.        4: invokespecial #3                  // Method java/lang/String."<init>":()V 
  15.        7: astore_1 
  16.        8new           #4                  // class TestMain$SubCaller 
  17.       11: dup 
  18.       12: invokespecial #5                  // Method TestMain$SubCaller."<init>":()V 
  19.       15: astore_2 
  20.       16: aload_2 
  21.       17: aload_1 
  22.       18: invokevirtual #6                  // Method TestMain$Caller.call:(Ljava/lang/String;)V 
  23.       21return 

正如上面的结果,18: invokevirtual #6 // Method TestMain$Caller.call:(Ljava/lang/String;)V这里是TestMain$Caller.call而非TestMain$SubCaller.call,因为编译期无法确定调用子类还是父类的实现,所以只能丢给运行时的动态绑定来处理。

当重载遇上重写

下面的例子有点变态哈,Caller类中存在call方法的两种重载,更复杂的是SubCaller集成Caller并且重写了这两个方法。其实这种情况是上面两种情况的复合情况。

下面的代码首先会发生静态绑定,确定调用参数为String对象的call方法,然后在运行时进行动态绑定确定执行子类还是父类的call实现。

  1. public class TestMain { 
  2.   public static void main(String[] args) { 
  3.       String str = new String(); 
  4.       Caller callerSub = new SubCaller(); 
  5.       callerSub.call(str); 
  6.   } 
  7.    
  8.   static class Caller { 
  9.       public void call(Object obj) { 
  10.           System.out.println("an Object instance in Caller"); 
  11.       } 
  12.        
  13.       public void call(String str) { 
  14.           System.out.println("a String instance in in Caller"); 
  15.       } 
  16.   } 
  17.    
  18.   static class SubCaller extends Caller { 
  19.       @Override 
  20.       public void call(Object obj) { 
  21.           System.out.println("an Object instance in SubCaller"); 
  22.       } 
  23.        
  24.       @Override 
  25.       public void call(String str) { 
  26.           System.out.println("a String instance in in SubCaller"); 
  27.       } 
  28.   } 

执行结果为

  1. 22:30 $ java TestMain 
  2. a String instance in in SubCaller 

验证

由于上面已经介绍,这里只贴一下反编译结果啦

  1. 22:30 $ javap -c TestMain 
  2. Compiled from "TestMain.java" 
  3. public class TestMain { 
  4.   public TestMain(); 
  5.     Code: 
  6.        0: aload_0 
  7.        1: invokespecial #1                  // Method java/lang/Object."<init>":()V 
  8.        4return 
  9.  
  10.   public static void main(java.lang.String[]); 
  11.     Code: 
  12.        0new           #2                  // class java/lang/String 
  13.        3: dup 
  14.        4: invokespecial #3                  // Method java/lang/String."<init>":()V 
  15.        7: astore_1 
  16.        8new           #4                  // class TestMain$SubCaller 
  17.       11: dup 
  18.       12: invokespecial #5                  // Method TestMain$SubCaller."<init>":()V 
  19.       15: astore_2 
  20.       16: aload_2 
  21.       17: aload_1 
  22.       18: invokevirtual #6                  // Method TestMain$Caller.call:(Ljava/lang/String;)V 
  23.       21return 

好奇问题

非动态绑定不可么?

其实理论上,某些方法的绑定也可以由静态绑定实现。比如

  1. public static void main(String[] args) { 
  2.       String str = new String(); 
  3.       final Caller callerSub = new SubCaller(); 
  4.       callerSub.call(str); 

比如这里callerSub持有subCaller的对象并且callerSub变量为final,立即执行了call方法,编译器理论上通过足够的分析代码,是可以知道应该调用SubCaller的call方法。

但是为什么没有进行静态绑定呢?
假设我们的Caller继承自某一个框架的BaseCaller类,其实现了call方法,而BaseCaller继承自SuperCaller。SuperCaller中对call方法也进行了实现。

假设某框架1.0中的BaseCaller和SuperCaller

  1. static class SuperCaller { 
  2.   public void call(Object obj) { 
  3.       System.out.println("an Object instance in SuperCaller"); 
  4.   } 
  5.    
  6. static class BaseCaller extends SuperCaller { 
  7.   public void call(Object obj) { 
  8.       System.out.println("an Object instance in BaseCaller"); 
  9.   } 

而我们使用框架1.0进行了这样的实现。Caller继承自BaseCaller,并且调用了super.call方法。

  1. public class TestMain { 
  2.   public static void main(String[] args) { 
  3.       Object obj = new Object(); 
  4.       SuperCaller callerSub = new SubCaller(); 
  5.       callerSub.call(obj); 
  6.   } 
  7.    
  8.   static class Caller extends BaseCaller{ 
  9.       public void call(Object obj) { 
  10.           System.out.println("an Object instance in Caller"); 
  11.           super.call(obj); 
  12.       } 
  13.        
  14.       public void call(String str) { 
  15.           System.out.println("a String instance in in Caller"); 
  16.       } 
  17.   } 
  18.    
  19.   static class SubCaller extends Caller { 
  20.       @Override 
  21.       public void call(Object obj) { 
  22.           System.out.println("an Object instance in SubCaller"); 
  23.       } 
  24.        
  25.       @Override 
  26.       public void call(String str) { 
  27.           System.out.println("a String instance in in SubCaller"); 
  28.       } 
  29.   } 

然后我们基于这个框架的1.0版编译出来了class文件,假设静态绑定可以确定上面Caller的super.call为BaseCaller.call实现。

然后我们再次假设这个框架1.1版本中BaseCaller不重写SuperCaller的call方法,那么上面的假设可以静态绑定的call实现在1.1版本就会出现问题,因为在1.1版本上super.call应该是使用SuperCall的call方法实现,而非假设使用静态绑定确定的BaseCaller的call方法实现。

所以,有些实际可以静态绑定的,考虑到安全和一致性,就索性都进行了动态绑定。

得到的优化启示?

由于动态绑定需要在运行时确定执行哪个版本的方法实现或者变量,比起静态绑定起来要耗时。

所以在不影响整体设计,我们可以考虑将方法或者变量使用private,static或者final进行修饰。

参考文章

一本书

  • Java核心技术,Java领域最有影响力和价值的著作之一,拥有20多年教学与研究经验的资深Java技术专家撰写(获Jolt大奖),与《Java编程思想》齐名,10余年全球畅销不衰,广受好评。

责任编辑:张伟 来源: 技术小黑屋
相关推荐

2012-01-09 11:26:15

Java

2009-07-22 08:52:05

Scala动态绑定

2011-08-22 09:34:50

Objective-C多态动态类型

2016-12-14 14:29:30

Java动态绑定机制

2014-04-02 09:21:52

2013-08-19 14:27:49

2016-09-27 16:26:58

2011-12-30 09:40:28

2017-08-07 16:39:03

JSX动态数据

2022-02-18 08:28:49

域名公网IP

2016-10-11 20:33:17

JavaScriptThisWeb

2010-07-30 10:45:08

Flex数据绑定

2021-12-12 20:10:49

域名动态IP

2011-03-30 09:13:13

静态类Windows Pho

2022-01-18 10:39:29

自动驾驶数据人工智能

2011-07-27 08:56:32

Oracle数据库绑定变量软解析

2023-10-20 09:51:00

编程开发

2023-10-07 11:04:58

WPF数据UI

2021-07-06 06:39:22

Java静态代理动态代理

2009-06-18 14:40:44

TreeView动态绑
点赞
收藏

51CTO技术栈公众号