概述:
下图为一张比较经典的jvm图(以hotspot虚拟机为例)
当一个类通过了加载,链接(验证,准备,解析),初始化以后,字节码文件被加载到方法区,这时候执行引擎(解释器或者是即时编辑器)就会执行对应的字节码指令(转成机器指令操作计算机)。这时候,就会用到了运行时数据区。
经典的运行时数据区(内存空间)包括:方法区(永久代或元空间),堆内存,虚拟机栈,本地方法栈,程序计数器(PC寄存器)
线程独有:程序计数器,虚拟机栈,本地方法栈
线程共享:堆,方法区(永久代或元空间)
jvm只有一个Runtime实例
jvm线程系统
Java程序启动以后,除了main方法及其自己创建的线程以外,还会有如下线程在运行:
1:虚拟机线程:这种线程的操作是需要JVM达到安全点才会出现。执行包括“stop-the-world”的垃圾收集,线程栈收集,线程挂起等
2:周期任务线程:用于周期性操作的调度
3:GC线程:对JVM里不同种类的垃圾收集行为提供支持
4:编译线程:这种线程在运行时会将字节码编译成本地执行代码指令
5:信号调度线程:接受信号,并且发送给JVM
6:等等····
当java线程准备好执行以后,操作系统的本地线程也同时创建,并将其调度到任何一个可用的CPU上,一旦本地线程初始化成功,它就会调用java线程中的run方法。
程序计数器
**作用:**用来存储指向下一条(当前正在执行)指令的地址,执行引擎会在这里读取下一条指令,然后执行,执行完,再将后面一条指令地址再更新进去
**特点:**空间很小,就存一个地址,运行速度也最快,一个线程一个,没有内存溢出,也不涉及垃圾回收
为了能够准确地记录各个线程正在执行的当前字节码指令地址,最好的办法是为每一个线程都分配一个PC寄存器,这样各个线程便可以独立运算,互不干扰。这也就是为啥它是线程私有的原因。