分派(Dispatch)可能是静态也可能是动态的,根据分派依据的宗量数可分为单分派和多分派。这两种分派方式的两两组合就构成了静态单分派,静态多分派,动态单分派,动态多分派这4种组合。本章讲静态分派。

1.静态分派

所有依赖静态类型来定位方法执行版本的分派动作称为静态分派。静态分派的典型应用是方法重载。静态分派发生在编译阶段,因此确定静态分派的动作实际上不是由虚拟机来执行的。

那么什么是静态类型(static type)呢?

  1. Super object = new Sub();

像上面的语句,Super是变量的静态类型,Sub是变量的实际类型(actual type),静态类型和实际类型在程序中都可以发生一些变化,区别是静态类型的变化仅仅在使用时发生,变量本身的静态类型不会被改变,并且最终的静态类型是在编译期可知的;而实际类型变化 的结果在运行期才可确定,编译器在编译程序的时候并不知道一个对象的实际地址是什么。

静态分派一词实际上是中文翻译特有的,国外的技术文档都是将其称为Method Overload Resolution。这样一来就更好理解了,因为是Resolution(解析)

下面的代码可以说明这一点:

  1. public static class Printer {
  2. public static void print(Super object) {
  3. System.out.println("it is Super");
  4. }
  5.  
  6. public static void print(Sub object) {
  7. System.out.println("it is Sub");
  8. }
  9. }

当调用print方法时,打印的将是"it is Super".

2. 调用“合适”的方法

编译器虽然能确定出方法的重载版本,但在很多情况下这个重载版本并不是“唯一的”,往往只能确定一个“更加合适”的版本。什么意思呢?看看下面的代码。

  1. public static void main(String[] args) {
  2. char c = 'a';
  3. Printer.print(c);
  4. }
  5.  
  6. public static class Printer {
  7.  
  8. public static void print(int i) {
  9. System.out.println("it is int");
  10. }
  11.  
  12. public static void print(byte b) {
  13. System.out.println("it is byte");
  14. }
  15. }

上面的代码可以执行吗?乍看之下,没有类型为char的重载方法,是不是会报错?实际上,会打印出 it is int。也就是说,虽然没有char类型参数的方法,但编译器通过参数自动转型帮你找到了一个“合适”的方法调用。

参数自动转型可参考 参数自动转型

转换的路径是char->int->long->float->double,如果还没找到合适的方法,则自动装箱成Character,此时已经是一个类。如果还找不到,则开始查找该类实现的接口(优先),父类(在继承关系中从下往上找,越接近上层的优先级越低)。如果有多个接口同时出现两个参数一致的,此时优先级是一样的,编译器无法确定自动转型为哪种类型,会提示类型模糊,拒绝编译。程序必须在调用时显式地指定字段的静态类型。

下面这个例子,没有参数为Sub的方法,按照参数自动转型,查找最合适方法的方式,会找到Super为参数方法调用。

  1. public static void main(String[] args) {
  2. Sub object = new Sub();
  3. Printer.print(object);
  4. }
  5.  
  6. public static class Printer {
  7. public static void print(Super object) {
  8. System.out.println("it is Super");
  9. }
  10. }

此外还要注意一点是传入参数为null. 如果重载方法里有两个不同的类型的参数,即使两者没有继承关系,编译器也会判断不了到底调用哪个。

  1. public static void main(String[] args) {
  2. Printer.print(null);
  3. }
  4.  
  5. public static class Printer {
  6. public static void print(Super object) {
  7. System.out.println("it is Super");
  8. }
  9.  
  10. public static void print(App app) {
  11. System.out.println("it is App");
  12. }
  13. }

在调用的时候强制转换,指定类型,就可以解决了。

  1. Printer.print((App)null);

要注意的一点是,解析与分派这两者之间的关系并不是二选一的排他关系,它们是在不同层次上去筛选,确定目标方法的过程。例如,静态方法在类加载期就会解析,但静态方法也是可以有重载版本的,选择重载版本的过程也是通过静态分派完成的。

JVM 方法调用之静态分派的更多相关文章

  1. JVM 方法调用之动态分派

    1. 动态分派 一个体现是重写(override).下面的代码,运行结果很明显. public class App { public static void main(String[] args) { ...

  2. JVM方法调用过程

    JVM方法调用过程 重载和重写 同一个类中,如果出现多个名称相同,并且参数类型相同的方法,将无法通过编译.因此,想要在同一个类中定义名字相同的方法,那么它们的参数类型必须不同.这种方法上的联系就是重载 ...

  3. JVM方法调用

    当我们站在JVM实现的角度去看方法调用的时候,我们自然会想到一种分类: 1.编译代码的时候就知道是哪个方法,永远不会产生歧义,例如静态方法,private方法,构造方法,super方法. 2.运行时才 ...

  4. JVM 方法调用之解析

    方法调用并不等同于方法执行,方法调用阶段唯一的任务就是确定被调用方法的版本(即调用哪一个方法),暂时还没有涉及到方法内部的具体运行过程.在程序运行时,进行方法调用是最普遍最频繁的操作,但Class文件 ...

  5. JVM学习三:静态分派

    所有依赖静态类型来定位方法的执行版本的分派动作成为静态分派,静态分派典型的应用场景是方法的重载.在编译阶段,javac编译器会根据参数的静态类型决定使用哪个重载版本,但很多种情况下这个版本并不是“唯一 ...

  6. Spring杂谈 | 从桥接方法到JVM方法调用

    前言 之所以写这么一篇文章是因为在Spring中,经常会出现下面这种代码 // 判断是否是桥接方法,如果是的话就返回这个方法 BridgeMethodResolver.findBridgedMetho ...

  7. JVM方法调用栈

    摘自深入分析java web技术内幕

  8. jvm 字节码执行 (一)方法调用

    “虚拟机”是一个相对于“物理机”的概念,这两种机器都有代码执行能力,其区别是物理机的执行引擎是直接建立在处理器.硬件.指令集和操作系统层面上,而虚拟机的执行引擎是 由自己实现的,因此可以自行制定指令集 ...

  9. java方法调用之动态调用多态(重写override)的实现原理——方法表(三)

    上两篇篇博文讨论了java的重载(overload)与重写(override).静态分派与动态分派.这篇博文讨论下动态分派的实现方法,即多态override的实现原理. java方法调用之重载.重写的 ...

随机推荐

  1. 第七章 API网关服务:Spring Cloud Zuul

    API网关是一个更为智能的应用服务器, 它的定义类似于面向对象设计模式中的Facade模式, 它的存在就像是整个微服务架构系统的门面一样,所有的外部客户端访问都需要经过它来进行调度和过滤.它除了要实现 ...

  2. Django学习笔记之Class-Based-View

    Django写的多了,有些问题才逐渐认识到. 比如有一个view比较复杂,调用了很多其他的函数.想要把这些函数封装起来,怎么办? 当然,可以用注释#------view------这样将函数隔离开,这 ...

  3. 打开当前目录的其他exe

    STARTUPINFO si; PROCESS_INFORMATION pi; memset(&si, , sizeof(si)); si.cb = sizeof(STARTUPINFO); ...

  4. WCF配置Tcp协议

    注意点: 1,<serviceMetadata httpGetEnabled="false"/>   2,       <services>         ...

  5. Python实现七牛云视频播放

    这篇文章是使用Python的Web框架Django Rest Framework来提供视频相关的api接口,主要功能包括视频上传.视频转码.视频访问授权.删除视频文件.视频截图功能. 七牛云上的基本概 ...

  6. Reducing File Size

    [Reducing File Size] 1.Unity strips out unused assets. The amount of assets in your project folder d ...

  7. spark属性

    应用属性 属性名 缺省值 意义 spark.app.name (none) The name of your application. This will appear in the UI and i ...

  8. MySQL 基础常用命令

    一.启动与关闭 1.1 Linux下启动mysql 的命令: a. rpm包安装:service mysqld start b. 源码包安装:/usr/local/mysql/bin/mysqld_s ...

  9. linux系统软件版本升级

    在安装完软件之后,在同一层目录生成一个符号链接,并把当前软件的目录映射到这个链接上,后面的操作都只通过这个链接去做,以后升级版本的时候,把最新的软件目录映射到这个链接上就可以了. 如我刚装的apach ...

  10. 本地yum源构建以及Docker离线安装

    Docker离线安装以及本地yum源构建 在docker的使用过程中有时候会遇到一些私有化部署的问题,就是在一些无法上网的机器上面安装使用dokcer,这就引出了docker的离线安装的问题,dock ...