字符串常量池不会自动膨胀,需同时满足显式intern、长期强引用、未被GC回收三条件;JDK 7+后移至堆中受GC管理;避免高频唯一字符串intern,合理配置StringTableSize与字符串去重。
字符串常量池不会自己“悄悄膨胀”,它只在特定条件下才可能积累大量对象,进而引发内存压力。关键不在于“会不会膨胀”,而在于“谁往里塞、怎么塞、塞了还留不留”。
常量池本身是受控区域,不是垃圾场。它膨胀必须同时满足三个条件:
intern()(比如循环中对动态生成的 new String("id_" + i) 反复 intern)这是最容易被误解的一点:JDK 6 及以前,字符串常量池在永久代(PermGen),GC 极少光顾,一旦塞满就容易 OOM: PermGen space;而从 JDK 7 开始,常量池被移到Java 堆中,完全纳入主流 GC 管理范围。
这意味着:
以下写法容易无意中制造常量池压力:
while (true) { String s = new StringBuilder().append(System.nanoTime()).toString().intern(); } —— 每次生成唯一时间戳并驻留,且无引用清理
intern 类名或资源路径注意:编译期字面量(如 "hello")天然入池,但数量可控、内容稳定,一般不构成威胁。
JDK 8u20+ 起,可通过参数启用两项关键优化:
-XX:+UseStringDeduplication(需配合 G1 GC):GC 过程中自动识别堆内重复字符串,共享底层 char[],大幅降低内存冗余-XX:StringTableSize=65536:手动增大字符串表桶数,减少哈希冲突,避免因扩容失败导致 intern 失败或性能下降这两项不是“救火措施”,而是预防性设计——尤其适合日志系统、序列化框架、模板引擎等高频字符串操作场景。
基本上就这些。常量池管理没那么玄,核心就三点:别乱 intern、别长期持引用、用对 JDK 版本和 GC 参数。