【本文是为了梳理知识的总结性文章,总结了一些自认为相关的重要知识点,只为巩固记忆以及技术交流,忘批评指正。其中参考了很多前辈的文章,包括图片也是引用,如有冒犯,侵删。】
对于Java中的常量池,一直有点困惑,看完网上的一些相关博客后,也似懂非懂,于是自己总结一下。关于常量池带着以下几个问题来逐步解答。
1 什么是常量池?
常量池毫无疑问是用来存放常量的,那么那些是常量呢?
常量池主要用于存放两大类常量:字面量和符号引用量。
- 字面量相当于Java语言层面常量的概念,如文本字符串,声明为final的常量值(成员变量)等。
- 符号引用则属于编译原理方面的概念,包括了如下三种类型的常量:
- 类和接口的全限定名
- 字段名称和描述符
- 方法名称和描述符
2 为什么需要常量池?
常量池是为了避免频繁的创建和销毁对象而影响系统性能,其实现了对象的共享。
例如字符串常量池,在编译阶段就把所有的字符串文字放到一个常量池中。常量池中所有相同的字符串常量被合并,只占用一个空间,节省了内存空间。
3 Java 中有哪些常量池?
Java中的常量池,实际上分为两种形态:静态常量池和运行时常量池。
- 静态常量池 : Class文件中的常量池存在于编译时期,Class文件中的常量池不仅仅包含字符串字面量,还包含类、方法的信息,占用Class文件绝大部分空间。
- 运行时常量池 :Jvm虚拟机在完成类装载操作后,在运行期将class文件中的常量池载入到内存中,并保存在方法区(JDK 8 中的元数据区)中,我们常说的常量池,就是指方法区(元数据区)中的运行时常量池。
此外还有字符串常量池,自认为也属于运行时常量池的一部分,不过和由于字符串的大量使用,于是单独进行管理。在JDK 1.6及以下,存放在PermGen Space中,而在 JDK 7和 JDK 8 中,存放到了堆内存中,并且 JDK 1.8中 PermSize 和 MaxPermGen 已经无效,使用元数据区替代了永久代。
补充 : 运行时常量池中的常量,基本来源于各个class文件中的常量池,即每个class文件都有对应的常量池。字符串常量池JVM 实例全局共享的,只有一个。
4 字符串常量池的使用
运行时常量池相对与Class文件常量池的一个重要特征就是具备动态性,Java 并不要求常量一定只有编译器才能产生,运行时可以起将新的常量放入池中。String的intern()方法就可以将某个String对象在运行期动态的加入字符串常量池(如果常量池中已经存在就不加)并返回String pool
中保证唯一的一个字符串对象的引用。