双引号创建的字符串自动入池,new创建的对象不入池;通过intern()可手动入池,提升重复字符串处理性能,但需注意内存管理。

在Java中,字符串常量池(String Pool)是JVM为了提高性能和节省内存而设计的一种机制。它存储所有字符串字面量的引用,避免重复创建相同内容的字符串对象。合理利用字符串常量池,可以显著减少内存占用并提升程序效率。
理解字符串常量池的工作机制
当使用双引号定义字符串时,例如 String str = "hello";,JVM会先检查字符串常量池中是否已存在内容相同的字符串。如果存在,就直接返回池中已有对象的引用;如果不存在,则在池中创建新对象并返回引用。
而通过 new String("hello") 的方式创建字符串,无论池中是否存在相同内容,都会在堆中新建一个String对象,仅其内部字符数组可能共享常量池中的值。
关键点:
立即学习“Java免费学习笔记(深入)”;
- 双引号创建的字符串自动入池
- new 创建的字符串对象不自动入池(除非调用intern)
- 字符串拼接中若含变量,通常不会直接进入常量池
使用intern()方法手动入池
对于运行时生成的字符串,可以通过调用 intern() 方法将其加入字符串常量池。该方法会检查池中是否已有相同内容的字符串,若有则返回池中引用;若无则将当前字符串复制到池中并返回其引用。
示例:
String s1 = new String("java");
String s2 = s1.intern();
String s3 = "java";
System.out.println(s2 == s3); // 输出 true
这个特性在处理大量重复字符串数据时非常有用,比如解析日志、读取配置文件或处理JSON字段名等场景。
避免不必要的字符串对象创建
在高频操作中,应尽量避免因字符串拼接导致的临时对象膨胀。例如,在循环中使用 += 拼接字符串会不断生成新的String对象,不仅绕过常量池,还加重GC负担。
建议:
- 使用 StringBuilder 或 StringBuffer 进行动态拼接
- 对最终结果调用 intern()(当确认内容高度重复时)
- 静态已知字符串尽量用字面量形式声明
若从数据库或网络获取大量相同标签、状态码等字符串,可在首次解析后调用 intern(),后续比较可直接用 == 判断,提升性能。
注意常量池的内存管理
字符串常量池在JDK7之后被移到堆内存中,不再位于永久代(PermGen),因此受堆大小限制,可通过 -Xmx 参数调节。但滥用 intern() 可能导致内存泄漏或OOM。
建议:
- 仅对高重复率的字符串调用 intern()
- 监控应用的字符串内存使用情况
- 避免对用户输入或唯一ID等低重复内容调用 intern()
基本上就这些。掌握字符串常量池的使用逻辑,能有效优化内存和性能,关键是理解创建方式与入池时机,结合实际场景合理选择处理策略。










