先了解下JVM中的内存分配,此处以hotspot vm为例(官方jdk采用的vm)
堆内存是各个线程共享的区域
它用于存储已经被虚拟机加载的类信息、常量、静态变量、即编译器编译后的代码等数据。静态变量、常量在方法区,所有方法,包括静态和非静态的,也在方法区
这里解释一下方法区:
首先方法区不是在堆中,在java8之前是用永久代实现的,永久代是 Hotspot 虚拟机特有的概念,虽然 Java 虚拟机规范把方法区描述为堆的一个逻辑部分,但是它却有一个别名叫做 Non-Heap(非堆),目的应该是与 Java 堆区分开来
java8之后是用元空间实现的,元空间并不在虚拟机中,而是使用本地内存。
只是 JVM 规范中定义的一个概念,用于存储被 JVM 加载的类信息、常量、静态变量、即时编译器编译后的代码等数据
常量池分为静态常量池和运行时常量池
也叫 class文件常量池,主要存放编译期生成的各种字面量(Literal)和符号引用(Symbolic References)。
final类型的常量在编译期间就会进行分配。
静态常量池指的是在编译期确定,保存在class文件中的一些数据。
静态常量池是在编译期间生成的,存储在.class文件中。
字面量(Literal)和符号引用量,字面量相当于Java语言层面常量的概念,如文本字符串、声明为final的常量值等,符号引用则属于编译原理方面的概念,包括了如下三种类型的常量:
1、类和接口的全限定名;
2、字段的名称和描述符;
3、方法的名称和描述符。
当类加载到内存中后,JVM就会将class常量池中的内容存放到运行时常量池中;运行时常量池里面存储的主要是编译期间生成的字面量、符号引用等等。
类加载在链接环节的解析过程,会符号引用转换成直接引用(静态链接)。此处得到的直接引用也是放到运行时常量池中的。
运行期间可以动态放入新的常量
运行时常量池内存位置:java8之前都在方法区中,java8之后在元空间
字符串常量池,也可以理解成运行时常量池分出来的一部分。类加载到内存的时候,字符串会存到字符串常量池里面。利用池的概念,避免大量频繁创建字符串
Hotspot8之前,使用持久代实现方法区,由于持久代内存不好估算,很容易到值OOM:Perm Gen异常。而元空间是本地内存,取决于操作系统分配内存。
Jdk1.6及之前: 有永久代, 运行时常量池在永久代,运行时常量池包含字符串常量池
Jdk1.7:有永久代,但已经逐步“去永久代”,字符串常量池从永久代里的运行时常量池分离到堆里
Jdk1.8及之后: 无永久代,运行时常量池在元空间,字符串常量池里依然在堆里
参考文章
(62条消息) 深入理解java虚拟机(全章节完整)_TJtulong的博客-CSDN博客
(62条消息) Java | JVM | 详细图解,坚持看完,带你真正搞懂Java虚拟机_唐 城的博客-CSDN博客
(62条消息) Java-双亲委派机制_七楼、的博客-CSDN博客
(62条消息) Java-类加载器11127222七楼、的博客-CSDN博客
(62条消息) jvm中的常量池到底在哪里?_常量池在哪里_yuan_qh的博客-CSDN博客
(62条消息) Java方法区和永久代_java永久代和方法区_yuan_qh的博客-CSDN博客
这一次,彻底弄懂java中的常量池 - 掘金 (juejin.cn)
(62条消息) JAVA常量池,一篇文章就足够入门了。(含图解)lei'chang'liang'chi河海哥yyds的博客-CSDN博客
手机扫一扫
移动阅读更方便
你可能感兴趣的文章