iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >从Java静态绑定和动态绑定中得到优化启示的示例分析
  • 672
分享到

从Java静态绑定和动态绑定中得到优化启示的示例分析

2023-06-17 07:06:00 672人浏览 安东尼
摘要

本篇文章给大家分享的是有关从Java静态绑定和动态绑定中得到优化启示的示例分析,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。一个Java程序的执行要经过编译和执行(解释)这两个

本篇文章给大家分享的是有关从Java静态绑定和动态绑定中得到优化启示的示例分析,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

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

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

区别对比

  • 静态绑定发生在编译时期,动态绑定发生在运行时

  • 使用private或static或final修饰的变量或者方法,使用静态绑定。而虚方法(可以被子类重写的方法)则会根据运行时的对象进行动态绑定。

  • 静态绑定使用类信息来完成,而动态绑定则需要使用对象信息来完成。

  • 重载(Overload)的方法使用静态绑定完成,而重写(Override)的方法则使用动态绑定完成。

重载方法的示例

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

public class TestMain {   public static void main(String[] args) {       String str = new String();       Caller caller = new Caller();       caller.call(str);   }    static class Caller {       public void call(Object obj) {           System.out.println("an Object instance in Caller");       }              public void call(String str) {           System.out.println("a String instance in in Caller");       }   } }

执行的结果为

22:19 $ java TestMain a String instance in in Caller

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

验证

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

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

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

重写方法的示例

public class TestMain {   public static void main(String[] args) {       String str = new String();       Caller caller = new SubCaller();       caller.call(str);   }      static class Caller {       public void call(String str) {           System.out.println("a String instance in Caller");       }   }      static class SubCaller extends Caller {       @Override       public void call(String str) {           System.out.println("a String instance in SubCaller");       }   } }

执行的结果为

22:27 $ java TestMain a String instance in SubCaller

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

验证

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

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

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

当重载遇上重写

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

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

public class TestMain {   public static void main(String[] args) {       String str = new String();       Caller callerSub = new SubCaller();       callerSub.call(str);   }      static class Caller {       public void call(Object obj) {           System.out.println("an Object instance in Caller");       }              public void call(String str) {           System.out.println("a String instance in in Caller");       }   }      static class SubCaller extends Caller {       @Override       public void call(Object obj) {           System.out.println("an Object instance in SubCaller");       }              @Override       public void call(String str) {           System.out.println("a String instance in in SubCaller");       }   } }

执行结果为

22:30 $ java TestMain a String instance in in SubCaller

验证

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

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

好奇问题

非动态绑定不可么?

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

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

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

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

假设某框架1.0中的BaseCaller和SuperCaller

static class SuperCaller {   public void call(Object obj) {       System.out.println("an Object instance in SuperCaller");   } }    static class BaseCaller extends SuperCaller {   public void call(Object obj) {       System.out.println("an Object instance in BaseCaller");   } }

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

public class TestMain {   public static void main(String[] args) {       Object obj = new Object();       SuperCaller callerSub = new SubCaller();       callerSub.call(obj);   }      static class Caller extends BaseCaller{       public void call(Object obj) {           System.out.println("an Object instance in Caller");           super.call(obj);       }              public void call(String str) {           System.out.println("a String instance in in Caller");       }   }      static class SubCaller extends Caller {       @Override       public void call(Object obj) {           System.out.println("an Object instance in SubCaller");       }              @Override       public void call(String str) {           System.out.println("a String instance in in SubCaller");       }   } }

然后我们基于这个框架的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静态绑定和动态绑定中得到优化启示的示例分析,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注编程网精选频道。

--结束END--

本文标题: 从Java静态绑定和动态绑定中得到优化启示的示例分析

本文链接: https://www.lsjlt.com/news/287674.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 从Java静态绑定和动态绑定中得到优化启示的示例分析
    本篇文章给大家分享的是有关从Java静态绑定和动态绑定中得到优化启示的示例分析,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。一个Java程序的执行要经过编译和执行(解释)这两个...
    99+
    2023-06-17
  • JavaScript动态绑定的示例分析
    这篇文章给大家分享的是有关JavaScript动态绑定的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。问题描述:假设我们的网页中动态生成了一个按钮,在这个按钮生成之前我们...
    99+
    2024-04-02
  • java静态和动态绑定的实例对比分析
    这篇文章主要介绍了java静态和动态绑定的实例对比分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇java静态和动态绑定的实例对比分析文章都会有所收获,下面我们一起来看看吧。不同绑定的比较静态绑定发生在编译期...
    99+
    2023-06-30
  • PHP后期静态绑定的示例分析
    小编给大家分享一下PHP后期静态绑定的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!基础知识1. 范围解析操作符 (::...
    99+
    2024-04-02
  • 如何理解Java多态中的动态绑定和静态绑定
    如何理解Java多态中的动态绑定和静态绑定,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。在多态中:成员变量和静态方法编译和运行都看左边;成员方法编译看左边,运行看右边,这是为什...
    99+
    2023-06-02
  • java中静态绑定和动态绑定有什么区别
    本篇文章为大家展示了java中静态绑定和动态绑定有什么区别,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。常用的java框架有哪些1.SpringMVC,Spring Web MVC是一种基于Java...
    99+
    2023-06-14
  • PHP中使用后期静态绑定的示例分析
    这篇文章给大家分享的是有关PHP中使用后期静态绑定的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。后期静态绑定在PHP中的使用什么叫后期静态绑定呢?其实我们在之前的文章PHP中的static中已经说过这个...
    99+
    2023-06-15
  • 后期静态绑定在PHP中的使用示例
    这篇“后期静态绑定在PHP中的使用示例”除了程序员外大部分人都不太理解,今天小编为了让大家更加理解“后期静态绑定在PHP中的使用示例”,给大家总结了以下内容,具有一定借鉴价值,内容详细步骤清晰,细节处理妥当,希望大家通过这篇文章有所收获,下...
    99+
    2023-06-06
  • Java中动态代理和静态代理的示例分析
    这篇文章主要介绍了Java中动态代理和静态代理的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。0、代理模式为什么要学习代理模式?这是SpringAOP的底层【Spri...
    99+
    2023-06-29
  • vue父子组件之模态框状态绑定的示例分析
    小编给大家分享一下vue父子组件之模态框状态绑定的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!日常开发中经常遇到的一个...
    99+
    2024-04-02
  • React中this绑定的示例分析
    这篇文章主要介绍了React中this绑定的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。 我们在re...
    99+
    2024-04-02
  • Linux动态库和静态库的示例分析
    今天就跟大家聊聊有关Linux动态库和静态库的示例分析,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。函数库分为静态库和动态库两种。创建Linux静态库和Linux动态库和使用它们在这...
    99+
    2023-06-16
  • Vue中数据绑定的示例分析
    这篇文章主要为大家展示了“Vue中数据绑定的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Vue中数据绑定的示例分析”这篇文章吧。为什么绑定简单的数据渲...
    99+
    2024-04-02
  • Angular10中双向绑定的示例分析
    这篇文章主要介绍了Angular10中双向绑定的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。将利用@Input()和@Output...
    99+
    2024-04-02
  • 微信小程序中实现动态绑定数据及动态事件处理的示例分析
    这篇文章主要为大家展示了“微信小程序中实现动态绑定数据及动态事件处理的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“微信小程序中实现动态绑定数据及动态事...
    99+
    2024-04-02
  • Vue中Class与Style绑定的示例分析
    这篇文章主要为大家展示了“Vue中Class与Style绑定的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Vue中Class与Style绑定的示例分析...
    99+
    2024-04-02
  • Angular4中HTML属性绑定的示例分析
    小编给大家分享一下Angular4中HTML属性绑定的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!简介基本HTML属性...
    99+
    2024-04-02
  • JavaScript中this绑定规则的示例分析
    这篇文章主要介绍JavaScript中this绑定规则的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!一、this 的绑定规则this 一共有 4 中绑定规则,接下来一一介绍...
    99+
    2024-04-02
  • AngularJS1.X中数据绑定的示例分析
    这篇文章将为大家详细讲解有关AngularJS1.X中数据绑定的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1、ng-bind<!DOCTYPE ...
    99+
    2024-04-02
  • js中DOM事件绑定的示例分析
    这篇文章主要为大家展示了“js中DOM事件绑定的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“js中DOM事件绑定的示例分析”这篇文章吧。js事件绑定J...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作