Java 常量池总结

【本文是为了梳理知识的总结性文章,总结了一些自认为相关的重要知识点,只为巩固记忆以及技术交流,忘批评指正。其中参考了很多前辈的文章,包括图片也是引用,如有冒犯,侵删。】

对于Java中的常量池,一直有点困惑,看完网上的一些相关博客后,也似懂非懂,于是自己总结一下。关于常量池带着以下几个问题来逐步解答。

1 什么是常量池?

常量池毫无疑问是用来存放常量的,那么那些是常量呢?

常量池主要用于存放两大类常量:字面量和符号引用量。

  • 字面量相当于Java语言层面常量的概念,如文本字符串,声明为final的常量值(成员变量)等。
  • 符号引用则属于编译原理方面的概念,包括了如下三种类型的常量:
  1. 类和接口的全限定名
  2. 字段名称和描述符
  3. 方法名称和描述符

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中保证唯一的一个字符串对象的引用。

【参考文献】:

  1. 深理解Java虚拟机
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中的内存分为栈、堆和常量池。 1. 栈(Stack):栈内存用于存储方法调用时的局部变量、方法参数、方法返回以及方法调用时的执行环境。栈是线程私有的,每个线程都有自己的栈空间。栈是一种后进先出(LIFO)的数据结构,它的内存管理自动进行,不需要手动分配和释放。当一个方法被调用时,会在栈上创建一个帧(Frame),帧包含了方法的局部变量和部分运行时数据。当方法执行完毕后,对应的帧会被销毁。 2. 堆(Heap):堆内存用于存储Java对象实例。堆是所有线程共享的一块内存区域。Java中通过关键字"new"来创建对象,对象会被分配在堆内存中。堆是一种动态分配和管理的内存区域,需要手动进行垃圾回收。Java虚拟机会负责对堆进行自动的内存分配和释放。 3. 常量池(Constant Pool):常量池用于存储字符串常量、类和接口的全限定名、字段和方法的名称和描述符等常量常量池是每个类或接口的一部分,在编译期间就被确定,并且保存在.class文件中。运行时,常量池的内容被加载到内存中的运行时常量池中,在程序执行过程中可以动态地添加、删除或修改常量池中的内容。 总结:栈用于方法调用和执行环境,堆用于存储对象实例,常量池用于存储字符串常量。它们在Java中扮演着不同的角色,并且具有不同的生命周期和管理方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值