zoukankan      html  css  js  c++  java
  • JVM

    总结

    • 静态分配,用于方法的重载
    • 动态分配,用于方法的多态

    请看蓝色的标记部分:

    • 当通过h1, h2去调用共同的接口方法时,就是常见的多态应用。这时是动态分配,看“实际类型 Actual Type”。
    • 当把h1, h2作为参数传给“重载的方法”时,这时是静态分配,要看“静态类型 Static Type”--Human,不再看“实际类型”了。

    静态分派(方法的重载)

    编译期间就确定了
    多见于方法的重载。

    “Human”称为变量的静态类型(Static Type),或者叫做的外观类型(Apparent Type),后面的“Man”则称为变量的实际类型(Actual Type),静态类型和实际类型在程序中都可以发生一些变化,区别是静态类型的变化仅仅在使用时发生,变量本身的静态类型不会被改变,并且最终的静态类型是在编译期可知的;而实际类型变化的结果在运行期才可确定,编译器在编译程序的时候并不知道一个对象的实际类型是什么。
    代码中定义了两个静态类型相同但实际类型不同的变量,但虚拟机(准确地说是编译器)在重载时是通过参数的静态类型而不是实际类型作为判定依据的。并且静态类型是编译期可知的,因此,在编译阶段,Javac编译器会根据参数的静态类型决定使用哪个重载版本,所以选择了sayHello(Human)作为调用目标。所有依赖静态类型来定位方法执行版本的分派动作称为静态分派。静态分派的典型应用是方法重载。静态分派发生在编译阶段,因此确定静态分派的动作实际上不是由虚拟机来执行的。

    动态分派(方法的重写,多态)

    运行期间才确定
    静态类型同样都是Human的两个变量man和woman在调用sayHello()方法时执行了不同的行为,并且变量man在两次调用中执行了不同的方法。导致这个现象的原因很明显,是这两个变量的实际类型不同。
    在实现上,最常用的手段就是为类在方法区中建立一个虚方法表。虚方法表不在方法区,也不在堆也不在栈里。虚拟机维护的,一个类只有一个方法表。虚方法表维护着各个方法的实际入口(方法区中的某个地址)。虚方法表中存放着各个方法的实际入口地址。如果某个方法在子类中没有被重写,那子类的虚方法表里面的地址入口和父类相同方法的地址入口是一致的,都指向父类的实现入口。如果子类中重写了这个方法,子类方法表中的地址将会替换为指向子类实现版本的入口地址。PPT图中,Son重写了来自Father的全部方法,因此Son的方法表没有指向Father类型数据的箭头。但是Son和Father都没有重写来自Object的方法,所以它们的方法表中所有从Object继承来的方法都指向了Object的数据类型。

    解析

    编译期间就确定了
    范围:类的静态方法,构造方法,私有方法。

    调用目标在程序代码写好、编译器进行编译时就必须确定下来。这类方法的调用称为解析。
    在Java语言中符合“编译期可知,运行期不可变”这个要求的方法,主要包括静态方法和私有方法两大类,前者与类型直接关联,后者在外部不可被访问,这两种方法各自的特点决定了它们都不可能通过继承或别的方式重写其他版本,因此它们都适合在类加载阶段进行解析。

  • 相关阅读:
    linux内核(四)内存管理单元MMU
    open函数详解
    linux内核(三)文件系统
    C++中数字与字符串之间的转换 scanf string总结(复习必读)
    hello程序的运行过程-从计算机系统角度
    剑指offer第12题打印从1到n位数以及大整数加法乘法
    2017-10-11第二次万革始面经
    为什么需要半关闭
    Ubuntu指令
    143. Reorder List
  • 原文地址:https://www.cnblogs.com/frankcui/p/14140880.html
Copyright © 2011-2022 走看看