【JVM】StringTable 字符串常量池

参考:javaGuide

字符串常量池 是 JVM 为了提升性能和减少内存消耗针对字符串(String 类)专门开辟的一块区域,主要目的是为了避免字符串的重复创建

String的不可变性

1.通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。

2.字符串常量池中是不会存储相同内容的字符串的。

String的内存分配

Java 6及以前,字符串常量池存放在永久代
Java 7 中 Oracle 的工程师对字符串池 的逻辑做了很大的改变,即将字符串常量池的位置调整到Java堆内

Java 8 中,字符串常量仍然在堆

图源:javaguide


为什么要改变位置?

主要是因为永久代(方法区实现)的 GC 回收效率太低 ,只有在整堆收集 (Full GC )的时候才会被执行 GC。Java 程序中通常会有大量的被创建的字符串等待回收,将字符串常量池放到堆 中,能够更高效及时地回收字符串内存。


intern() 方法与 new String()

String s1 = new String("abc"); 这句话创建了几个字符串对象?

会创建 1 或 2 个字符串对象。

1、如果字符串常量池中不存在字符串对象"abc"的引用 ,那么它会在堆上创建两个字符串对象,其中一个字符串对象的引用会被保存在字符串常量池中。

2、如果字符串常量池中已存在字符串对象"abc"的引用 ,则只会在堆中创建 1 个字符串对象"abc"。


String.intern() 是一个 native(本地)方法,其作用是将指定的字符串对象的引用保存在字符串常量池 中,可以简单分为两种情况:

● 如果字符串常量池中保存了对应的字符串对象的引用,就直接返回该引用。

● 如果字符串常量池中没有保存对应的字符串对象的引用,那就在常量池中创建一个指向该字符串对象的引用并返回。

StringTable数据结构(拓展)

具体细节:数组+链表

  • String的String Pool是一个固定大小的Hashtable保存的是字符串(key)和 字符串对象的引用(value)的映射关系),默认值大小长度是1009,如果放进String Pool的String非常多,就会造成Hash冲突严重,从而导致链表会很长,而链表长了后直接会造成的影响就是当调用String.intern时性能会大幅下降(因为要一个一个找)。
  • jdk6 中StringTable是固定的 ,就是1009的长度,所以如果常量池中的字符串过多就会导致效率下降很快。在jdk7中 ,StringTable的长度可以通过一个参数指定:
    -XX:StringTableSize=99991
相关推荐
二哈赛车手8 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
栗子~~8 小时前
JAVA - 二层缓存设计(本地缓冲+redis缓冲+广播所有本地缓冲失效) demo
java·redis·缓存
YDS8298 小时前
DeepSeek RAG&MCP + Agent智能体项目 —— RAG知识库的搭建和接口实现
java·ai·springboot·agent·rag·deepseek
未若君雅裁10 小时前
MyBatis 一级缓存、二级缓存与清理机制
java·缓存·mybatis
AI人工智能+电脑小能手10 小时前
【大白话说Java面试题 第65题】【JVM篇】第25题:谈谈对 OOM 的认识
java·开发语言·jvm
阿维的博客日记11 小时前
Nacos 为什么能让配置动态生效?(涉及 @RefreshScope 注解)
java·spring
雨辰AI11 小时前
SpringBoot3 + 人大金仓读写分离 + 分库分表 + 集群高可用 全栈实战
java·数据库·mysql·政务
辰海Coding12 小时前
MiniSpring框架学习-完成的 IoC 容器
java·spring boot·学习·架构
小小编程路12 小时前
C++ 多线程与并发
java·jvm·c++
AI视觉网奇12 小时前
linux 检索库 判断库是否支持
java·linux·服务器