String中intern的方法

本文详细解析了 Java 中 String 类的 intern 方法的工作原理,包括如何使用 intern 方法来优化字符串处理,以及 intern 方法在字符串池中的行为表现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载自:http://www.cnblogs.com/wanlipeng/archive/2010/10/21/1857513.html 感谢分享!

首先查看官方API那个的解释:

———————————————————————————————————————

intern

public String intern()

返回字符串对象的规范化表示形式。

一个初始时为空的字符串池,它由类 String 私有地维护。

当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。

它遵循对于任何两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。

所有字面值字符串和字符串赋值常量表达式都是内部的。

返回:

一个字符串,内容与此字符串相同,但它保证来自字符串池中。

 

———————————————————————————————————————

尽管在输出中调用intern方法并没有什么效果,但是实际上后台这个方法会做一系列的动作和操作。在调用”ab”.intern()方法的时候会返回”ab”,但是这个方法会首先检查字符串池中是否有”ab”这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。

可以看下面一个范例:

复制代码
  
1 String str1 = " a " ;
2 String str2 = " b " ;
3 String str3 = " ab " ;
4 String str4 = str1 + str2;
5 String str5 = new String( " ab " );
6
7 System.out.println(str5.equals(str3));
8 System.out.println(str5 == str3);
9 System.out.println(str5.intern() == str3);
10 System.out.println(str5.intern() == str4);
复制代码

得到的结果:

true

false

true

false

 

为什么会得到这样的一个结果呢?我们一步一步的分析。

第一、str5.equals(str3)这个结果为true,不用太多的解释,因为字符串的值的内容相同。

第二、str5 == str3对比的是引用的地址是否相同,由于str5采用new String方式定义的,所以地址引用一定不相等。所以结果为false。

第三、当str5调用intern的时候,会检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用。

第四,当str4 = str1 + str2后,str4的值也为”ab”,但是为什么这个结果会是false呢?先看下面代码:

复制代码
  
1 String a = new String( " ab " );
2 String b = new String( " ab " );
3 String c = " ab " ;
4 String d = " a " + " b " ;
5 String e = " b " ;
6 String f = " a " + e;
7
8 System.out.println(b.intern() == a);
9 System.out.println(b.intern() == c);
10 System.out.println(b.intern() == d);
11 System.out.println(b.intern() == f);
12 System.out.println(b.intern() == a.intern());
复制代码

运行结果:

false

true

true

false

true

由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。

    当研究到这个地方的时候,突然想起来经常遇到的一个比较经典的Java问题,就是对比equal和==的区别,当时记得老师只是说“==”判断的是“地址”,但是并没说清楚什么时候会有地址相等的情况。现在看来,在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用。

执行下面的代码:

复制代码
  
1 String a = " abc " ;
2 String b = " abc " ;
3 String c = " a " + " b " + " c " ;
4 String d = " a " + " bc " ;
5 String e = " ab " + " c " ;
6
7 System.out.println(a == b);
8 System.out.println(a == c);
9 System.out.println(a == d);
10 System.out.println(a == e);
11 System.out.println(c == d);
12 System.out.println(c == e);
复制代码

运行的结果:

true

true

true

true

true

true

运行的结果刚好验证了我刚才的猜想。


### Java 中 `String` 的 `intern()` 方法工作原理 在 Java 中,`String.intern()` 是一种机制,用于将字符串实例存储到 JVM 的 **字符串常量池** (String Constant Pool) 中。如果该字符串已经存在于常量池中,则会返回其对应的引用;否则,当前字符串会被添加至常量池,并返回新加入的引用[^1]。 #### 原理分析 当调用 `s.intern()` 时,JVM 首先会在字符串常量池中查找是否存在与 `s` 内容相同的字符串。如果有匹配项,则直接返回已有字符串的引用;如果没有找到,则将此字符串复制一份存入字符串常量池并返回新的引用[^3]。 以下是实现这一功能的核心逻辑: ```java // 示例代码展示 intern() 方法的行为 public class InternExample { public static void main(String[] args) { String s1 = new String("hello").intern(); // 调用了 intern() String s2 = "hello"; // 直接从常量池加载 System.out.println(s1 == s2); // 输出 true,因为两者指向同一对象 } } ``` 上述代码片段展示了如何通过 `intern()` 将堆上的字符串移入字符串常量池,并验证两个变量是否共享同一个引用[^4]。 --- ### 使用场景 `String.intern()` 主要适用于以下几种情况: 1. **节省内存空间** 当程序中有大量重复的字符串数据时,可以利用 `intern()` 来减少冗余的对象创建,从而降低内存消耗[^2]。 2. **提高比较效率** 如果频繁执行基于内容的字符串比较操作(如使用 `.equals()`),可以通过 `intern()` 和身份比较 (`==`) 提升性能,因为后者通常比前者更快。 3. **处理固定集合的数据** 对于一些固定的字典或者枚举类型的字符串列表,提前将其放入字符串常量池有助于后续快速检索和判断。 --- ### 注意事项 尽管 `String.intern()` 可以为特定应用场景带来好处,但也需要注意潜在的风险或局限性: - 过度依赖可能导致不必要的复杂性和维护成本; - 在某些版本的 JDK 下可能会引发垃圾回收压力增加等问题。 因此,在实际开发过程中应权衡利弊后再决定是否采用这种方法来管理字符串资源。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值