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

举报
韩曙亮 发表于 2022/01/14 00:25:41 2022/01/14
【摘要】 文章目录 一、线程栈二、栈帧三、栈帧 - 局部变量表四、反汇编字节码文件五、Java 虚拟机指令手册六、程序计数器 一、线程栈 装载 HelloWorld.clas...





一、线程栈



装载 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();
    }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

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

java HelloWorld.class

  
 
  • 1

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

public static void main(String[] args) {

  
 
  • 1

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


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





二、栈帧



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

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

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

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


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





三、栈帧 - 局部变量表



局部变量表 :

以如下方法为例 :

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

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

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


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

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

  
 
  • 1
  • 2
  • 3
  • 4




四、反汇编字节码文件



使用

javac HelloWorld.java

  
 
  • 1

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

在这里插入图片描述


使用

javap -c HelloWorld.class

  
 
  • 1

命令 , 对 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
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33

在这里插入图片描述





五、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

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12

文章来源: hanshuliang.blog.csdn.net,作者:韩曙亮,版权归原作者所有,如需转载,请联系作者。

原文链接:hanshuliang.blog.csdn.net/article/details/120055452

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。