Java 虚拟机原理线程栈 | 栈帧 | 局部变量表 | 反汇编字节码文件 | Java 虚拟机指令手册 | 程序计数器

Posted 韩曙亮

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java 虚拟机原理线程栈 | 栈帧 | 局部变量表 | 反汇编字节码文件 | Java 虚拟机指令手册 | 程序计数器相关的知识,希望对你有一定的参考价值。





一、线程栈



装载 HelloWorld.class 字节码文件到 Java 虚拟机内存中 , 会将该字节码文件中的数据进行分解 , 放到不同的内存区域中 ;

public class HelloWorld {

    public int add() {
        int a = 1;
        int b = 1;
        int c = a + b;
        return c;
    }

    public static void main(String[] args) {
        HelloWorld helloWorld = new HelloWorld();
        helloWorld.add();
    }
}

运行该 HelloWorld.class 字节码文件 , 会创建一个进程 ;

java HelloWorld.class

main 方法是程序入口 , 运行后会创建一个线程 , 就是程序的主线程 ;

public static void main(String[] args) {

此时会针对 main 主线程 , 创建主线程的线程栈 ;


每个 线程 , 都要创建一个 线程栈 ;





二、栈帧



创建 main 主线程独有的 线程栈 , 主要存放 " 栈帧 " , 每个方法都对应一个 栈帧 , 这里存放的是 main 方法对应的栈帧 , 栈帧中存放 临时变量 , 操作数 ;

" 栈帧 " 同数据结构中的 栈 性质相同 , 先进后出 , 后入先出 ;

主线程 线程栈 中 , 执行 main 函数 , 放入了 main 方法的 栈帧 , 然后创建了 HelloWorld 对象 , 又执行该对象的 add 方法 , 又放入了 add 方法的 栈帧 ;

线程栈 中以 栈 的方式 管理 " 栈帧 " , 后进入的 栈帧 先执行 , 执行完毕后 , 从 线程栈 中 移出 ;


" 栈帧 " 中存储的是 局部变量表 , 操作数栈 , 动态链接 , 方法出口 ;





三、栈帧 - 局部变量表



局部变量表 :

以如下方法为例 :

    public int add() {
        int a = 1;
        int b = 1;
        int c = a + b;
        return a + b;
    }

局部变量表 , 存储局部变量 , 就是上述方法中的 a , b, c , 3 3 3 个局部变量 ;


在 main 方法的 栈帧 的局部变量表中 , 存储局部变量 helloWorld ; 但是注意 HelloWorld 对象的数据存储位置是 堆 ;

    public static void main(String[] args) {
        HelloWorld helloWorld = new HelloWorld();
        helloWorld.add();
    }




四、反汇编字节码文件



使用

javac HelloWorld.java

命令 , 将 HelloWorld.java 编译为 HelloWorld.class 字节码文件


使用

javap -c HelloWorld.class

命令 , 对 HelloWorld.class 字节码文件进行反汇编 ;

D:\\java>javap -c HelloWorld.class
Compiled from "HelloWorld.java"
public class HelloWorld {
  public HelloWorld();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public int add();
    Code:
       0: iconst_1
       1: istore_1
       2: iconst_1
       3: istore_2
       4: iload_1
       5: iload_2
       6: iadd
       7: istore_3
       8: iload_3
       9: ireturn

  public static void main(java.lang.String[]);
    Code:
       0: new           #2                  // class HelloWorld
       3: dup
       4: invokespecial #3                  // Method "<init>":()V
       7: astore_1
       8: aload_1
       9: invokevirtual #4                  // Method add:()I
      12: pop
      13: return
}





五、Java 虚拟机指令手册



反汇编的结果都是 Java 虚拟机指令 ; 这些指令都是交给 Java 虚拟机 执行的 ;

根据 Java 虚拟机 指令手册 , 分析上面的 Java 虚拟机指令 ;

附件中有一份 Java 虚拟机指令手册 , 可以在博客资源中下载 ;





六、程序计数器



CPU 时间片轮转 : 假设有一个单核 CPU , 给每个线程都会划分一个运行时间 ;

将 1 秒钟拆分成 1000 份 , 每份 1ms ; 很多线程在争取 CPU 资源 , 操作系统需要给每个线程进行 CPU 时间分配 , 如给线程 1 分配 3ms , 线程 2 分配 5ms , 线程 1 执行完毕后 , 马上切换到线程 2 执行 , 线程 2 执行完毕后 , 马上其它线程继续抢占 ;


线程 1 执行了 3ms , 然后 CPU 运行线程 2 , 假如 x ms 之后 , 再次回到线程 1 运行 , 需要靠程序计数器记录应该执行哪条 JVM 指令 ;


多个线程并发运行时 , 相互交叉抢占 CPU 资源 , 线程执行完分配的 CPU 时间后 , 需要记录下当前运行到哪 , 下一次分配到 CPU 资源后 , 继续执行哪条 JVM 指令 , 这里就需要 程序计数器 来实现该功能 ;


程序计数器就是记录下面的 JVM 指令前的数字 ;

  public int add();
    Code:
       0: iconst_1
       1: istore_1
       2: iconst_1
       3: istore_2
       4: iload_1
       5: iload_2
       6: iadd
       7: istore_3
       8: iload_3
       9: ireturn

以上是关于Java 虚拟机原理线程栈 | 栈帧 | 局部变量表 | 反汇编字节码文件 | Java 虚拟机指令手册 | 程序计数器的主要内容,如果未能解决你的问题,请参考以下文章

Java 内存结构之虚拟机栈

java虚拟机内存区常用名词解释

JVM学习笔记四:运行时数据区之虚拟机栈

JVM:Java虚拟机栈概述栈帧栈帧中的局部变量表

Java8虚拟机内存模型

Java虚拟机栈