在整个JVM中,程序计数器就如同操作系统的计数器一般,永远指明我们要去的方向
1.介绍
JVM中的程序计数寄存器(Program Counter Register)中,Register起源于CPU的寄存器,寄存器保存指令的现场情况。JVM中的PC寄存器是对物理PC寄存器的一种抽象模拟。
程序计数器是一块很小的内存空间,也是运行速度最快的存储区域,在JVM中,每个进程有若干个线程,每个线程都有自己的程序计数器,是线程私有的,它的生命周期伴随着整个线程
2. 作用
PC寄存器用来存储指向下一条指令的地址,即将要执行的指令代码。由存储引擎读取下一条指令。
- 它是程序控制流的指示器,分支,循环,跳转,异常处理等基础功能 ,都需要依赖这个程序计数器来完成
- 字节码解释器工作时,就是通过读取程序计数器的值来选择下一条需要执行的字节码指令
3. 举例说明
public class PCRegistryTest {
//javap -verbose PCRegistryTest.class 在out目录下反编译
public static void main(String[] args) {
int i = 10;
int j = 20;
int k = i + j;
}
}
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=4, args_size=1
// 指令地址:操作指令
0: bipush 10 //入栈
2: istore_1 //将其存入局部变量表
3: bipush 20 //入栈
5: istore_2
6: iload_1
7: iload_2
8: iadd
9: istore_3
10: return
上述的0,2,3,5等等就是程序计数器中存储的指令地址(0地址),根据这些指令地址,就可以执行当前方法中对应的操作指令,执行当前方法
并且可以从指令看出,为什么说Java的实现是使用栈的方式实现的。
4. 两个常见的问题
4.1 使用PC寄存器存储字节码指令地址有什么用?为什么要使用PC寄存器记录当前线程的执行地址呢?
因为CPU需要不停的切换各个线程,使用PC寄存器就可以清楚接着从哪开始继续执行。
JVM的字节码解释器需要通过改变PC寄存器的值来明确下一条应该执行什么样的字节码指令。
4.2 PC寄存器为什么是线程私有的?
在多线程的情况下,一个特定的时间段内只会执行某一个线程的方法,CPU会不停的做任务轮换,这样就必然导致中断和恢复,如何保证恢复后从上一次中断的地方继续执行呢?
为了能够准确的记录各个线程正在执行的当前字节码指令地址,最好的方法自然就是给每一个线程都分配一个PC寄存器
,这样各线程之间便可以相互进行独立计算,从而不会出现相互干扰的情况。