JVM 程序计数器

2023-09-21 01:13:13

Java虚拟机(JVM, Java Virtual Machine)是一个能够执行Java字节码的虚拟机。在JVM的架构中,程序计数器(Program Counter, PC)是一个关键的组成部分。程序计数器用于存储当前正在执行的Java字节码指令的地址。

每个线程在JVM中都有自己的程序计数器。

每条线程都需要有一个独立的程序计数器,各条线程之间计数器互不影响,独立存储,我们称这类内存区域为“线程私有”的内存

如果线程正在执行的是一个Java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是本地(Native)方法,这个计数器值则应为空(Undefined)

作用:

  1. 指令跟踪: 程序计数器记录了即将要执行的下一条指令的地址。这样,JVM就可以知道接下来需要执行哪一条指令。

  2. 线程隔离: 由于每个线程都有自己的程序计数器,当多个线程并发执行时,各个线程之间的执行流程不会互相干扰。

  3. 异常处理: 当异常发生时,程序计数器的值可以被用于确定异常发生的位置,从而有助于异常处理机制恢复或者处理错误。

工作原理

  1. 初始化: 当一个线程启动并开始执行一个方法时,程序计数器会被初始化为这个方法的第一条字节码指令的地址。

  2. 指令读取和执行: JVM根据程序计数器当前指向的地址,从方法的字节码中读取并执行相应的指令。

  3. 更新: 执行完一条字节码指令后,程序计数器的值会自动更新,以指向下一条需要执行的字节码指令。

  4. 方法调用和返回: 当发生方法调用时,程序计数器会被设置为被调用方法的第一条指令的地址。当方法执行完毕并返回时,程序计数器的值会被恢复为调用方法中的下一条指令地址。

  5. 异常处理: 如果在执行字节码指令过程中发生异常,程序计数器可以用于确定哪一条指令导致了异常,以便进行后续的异常处理。

  6. 线程切换: 在多线程环境下,当一个线程被挂起(例如,由于时间片用完或等待某个资源),其当前的程序计数器的值会被保存。当这个线程再次被调度执行时,程序计数器会被恢复到之前保存的值,从而使线程能够从中断的地方继续执行。

举个简单的例子,假设有以下的Java代码:

public class HelloWorld {
    public static void main(String[] args) {
        int x = 10;
        int y = 20;
        int sum = x + y;
        System.out.println("Sum is: " + sum);
    }
}

当这个程序运行在JVM上时:

  • 程序计数器首先会指向main方法中第一条字节码指令(通常是初始化x的指令)。
  • 执行这条指令后,程序计数器会更新,指向下一条指令(初始化y)。
  • 这个过程会一直持续,直到main方法执行完毕。
  • 如果有方法调用(例如System.out.println),程序计数器会暂时保存main方法当前的状态,然后跳转到新方法的字节码。

通过这种方式,程序计数器在JVM内部起到了关键的作用,使得JVM能够正确、高效地执行Java字节码。

Java方法和Native方法时的行为。


1. 执行Java方法
考虑以下简单的Java代码:


public class Calculator {
    public static int add(int a, int b) {
        return a + b;
    }

    public static void main(String[] args) {
        int result = add(5, 3);
        System.out.println("The result is: " + result);
    }
}

在这段代码的执行过程中,程序计数器的行为如下:

1.初始化:线程启动,程序计数器初始化为main方法的第一条指令的地址。
2.方法调用:当main方法调用add方法时,程序计数器记录了跳转到add方法的第一条指令的地址。
3.执行Java方法:程序计数器会依次指向add方法中各条指令的地址,直到这个方法执行完成。
4.返回和继续执行:add方法执行完毕后,程序计数器返回到main方法,指向调用add方法后的下一条指令的地址,然后继续执行。

2. 执行Native方法
Java允许调用Native方法,即使用Java Native Interface (JNI) 编写的本地方法。以下是一个示例:


public class NativeExample {
    static {
        System.loadLibrary("nativeLib"); // Load the native library
    }

    public native int nativeAdd(int a, int b);

    public static void main(String[] args) {
        NativeExample example = new NativeExample();
        int result = example.nativeAdd(5, 3);
        System.out.println("The result is: " + result);
    }
}

与之对应的C语言实现:


#include <jni.h>
#include "NativeExample.h"

JNIEXPORT jint JNICALL Java_NativeExample_nativeAdd(JNIEnv *env, jobject obj, jint a, jint b) {
    return a + b;
}

在这段代码的执行中,程序计数器的行为如下:

1.初始化:线程启动,程序计数器初始化为main方法的第一条指令的地址。
2.Native方法调用:当main方法调用nativeAdd方法时,程序计数器的值变为undefined。
3.Native方法执行:程序计数器的值保持undefined,因为在本地方法执行期间,JVM不控制执行流程。
4.返回和继续执行:当Native方法执行完成并返回到Java方法时,程序计数器的值会被恢复。

更多推荐

CSS 布局 (三) 浮动、定位、多列布局

6、浮动最初用于在文本块内浮动图像,float属性成为在网页上创建多列布局最常用的工具之一。随着flexbox和grid的出现,它现在又回到了最初的目的,正如本文所解释的那样。6.1浮动的背景引入float属性是为了允许web开发人员实现包含图像在文本列内浮动的布局,文本在其左侧或右侧环绕。就像你在报纸版面上看到的那样

函数扩展之——内存函数

前言:小伙伴们又见面啦。本篇文章,我们将讲解C语言中比较重要且常用的内存函数,并尝试模拟实现它们的功能。让我们一起来学习叭。目录一.什么是内存函数二.内存函数有哪些1.memcpy(1)库函数memcpy(2)模拟实现memcpy2.memmove(1)库函数memmove(2)模拟实现memmove3.memset4

【大数据】HDFS 的常用命令

HDFS的常用命令1.操作命令1.1创建文件夹1.2列出指定的文件和目录1.3新建文件1.4上传文件1.5将本地文件移动到HDFS1.6下载文件1.7查看文件1.8追写文件1.9删除目录或者文件1.10显示占用的磁盘空间大小1.11HDFS中的文件复制1.12HDFS中的文件移动2.管理命令2.1报告文件系统的基本信息

ElasticSearch(二)

1.DSL查询文档elasticsearch的查询依然是基于JSON风格的DSL来实现的。1.1.DSL查询分类Elasticsearch提供了基于JSON的DSL(DomainSpecificLanguage)来定义查询。常见的查询类型包括:查询所有:查询出所有数据,一般测试用。例如:match_all全文检索(fu

高教杯数学建模A题程序设计要点与思路

2023年是我最后一次参加高教杯大学生数学建模竞赛以后不会再参加了(大四参加意义不太,研究生有研究生的数学建模大赛)很遗憾由于各种原因我们没有能够完成赛题2022年美赛2022年MathorCup2022年国赛2022亚太杯2023年美赛2023年国赛我和我的朋友一共参加了6次比赛6次比赛我交到了很好的朋友然鹅成绩比较

Java 8 新特性解读及应用实践

Java8新特性解读及应用实践一、简介二、Lambda表达式三、流式编程四、日期/时间API1.概述2.LocalDate、LocalTime、LocalDateTime等类的使用3.格式化与解析五、重复注解和类型注解1.概念与作用2.重复注解实例3.类型注解实例六、小结回顾一、简介Java8带来了众多重大改进和新特性

【网络协议】Http-上

Http请求结构:结构图1:实验解析请求报文:1.在Edge浏览器上输入ip地址+端口号+文件资源,也就是下图中的120.XX.139.29:8888/A/B/c.html2.我的程序接收到了一个没有有效载荷的http请求(呼应上面的结构图1),如下GET/1/2/3.htmlHTTP/1.1//请求行(请求方法+请求

关于路由懒加载

路由懒加载是一种优化技术,它是可以延迟加载应用程序的某些模块或者组件,而不是在初始加载时一次性加载所有内容,这样可以减少初始加载的文件体积,提高应用程序的加载速度1.懒加载是在什么时候加载路由懒加载是在用户访问相应的路由时才进行加载。它延迟加载路由组件,使得在初始加载时只加载必要的代码,而将其他路由的代码推迟到需要时再

实验篇——根据群体经纬度提取环境数据(先导)

实验篇——根据群体经纬度提取环境数据(先导)文章目录前言一、获取数据文件1.1.经纬度文件1.2.环境数据的tif文件二、R语言代码实现三、环境文件介绍3.1.bio3.2.prec3.3.elev3.4.tmin3.5.tmax3.6.vapr13.7.tavg3.8.srad3.9.wind四、后续总结前言首先得到

LLM - 大模型技术报告与训练细节 By Baichuan2

目录一.引言二.Introduction-LLM相关进展1.模型参数越大,模型能力越强2.开源模型促进LLM领域快速发展3.开源模型集中在英文领域,其他语言能力有限4.训练数据2.6亿Token遥遥领先5.优化人类指令发布对应Chat模型6.公布了训练过程中的CKPT促进领域研究发展三.Pre-training-Bai

Vue路由与nodejs环境搭建

目录一、Vue路由1.1SPA简介1.2路由简介1.3路由实现思路1.3.1引入vue-router的js依赖1.3.2定义组件1.3.3定义路由1.3.4组装路由器1.3.5将路由挂载根实例1.3.6定义触发路由的按钮1.3.7定义锚点1.4示例二、nodejs环境搭建2.1nodejs简介2.2nodejs下载2.

热文推荐