静态变量所引用的实例位于Java堆或运行时常量池。

Java字节码与Native机器码不同,字节码是运行在JVM这一平台上的,字节码在被解释的过程中,具体的执行方式因JVM的不同实现而不同,但是对于JVM来说,它的各种不同实现都必须要遵循Java虚拟机规范。JVM的运行时数据区包含以下部分:

1、PC寄存器

每一条Java虚拟机线程都有自己的PC寄存器,如果正在被线程执行的当前方法不是native的,那么PC寄存器就保存Java虚拟机正在执行的字节码指令的地址。

2、Java虚拟机栈

每一条Java虚拟机线程都有自己私有的Java虚拟机栈,这个栈与线程同时创建,用于存储栈帧,栈帧可以在堆中分配,Java虚拟机栈所使用的内存不需要保证是连续的。

3、Java堆

在Java虚拟机中,堆是可供各个线程共享的运行时内存区域,也是供所有类实例和数组对象分配内存的区域,存储了被垃圾收集器所管理的各种对象。

4、方法区

在Java虚拟机中,方法区是可供各个线程共享的运行时内存区域,它存储了每一个类的结构信息。虽然方法区是堆的逻辑组成部分,但是简单的虚拟机实现可以选择在这个区域不实现垃圾收集与压缩。从Java6至Java14的虚拟机规范中,都没有限定实现方法区的内存位置和编译代码的管理策略。比如在HotSpot曾经的实现中,它内部的垃圾收集器全都基于“经典分代”来设计,将堆内存划分为新生代、老年代、永久代,其中永久代便是包括类型信息、常量、静态变量、JIT代码缓存等数据的方法区,而到了Java8,HotSpot已经完全废弃了永久代的概念,改用本地内存中实现的元空间来代替。

5、运行时常量池

运行时常量池的功能类似于ELF中的符号表,但是运行时常量池的包含的范围比典型的符号表大。一般来说,除了保存class文件中描述的符号引用外,还会把由符号引用翻译出来的直接引用也存储在运行时常量池中。相对于Class文件常量池来说,运行时常量池具备一个重要特性——动态性,并非预置入Class文件中常量池的内容才能进入运行时常量池,运行期间也可以将新的常量放入池中,比如String的intern方法。

6、本地方法栈

为虚拟机使用到的本地方法服务,具体的虚拟机可以根据需要自由实现它。

根据Java虚拟机规范的限制,由于静态变量所引用的实例可以被各个线程所共享,那么它一定不能位于PC寄存器、Java虚拟机栈、本地方法栈,又由于方法区存储的是类的结构信息而不是实例数据,所以被静态变量所引用的实例一定位于Java堆或运行时常量池。

Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐