java-----String的intern关键字

本文详细对比了JDK1.6与JDK1.7中String.intern()方法的行为差异,特别是在字符串常量池中的处理方式,并通过代码示例验证了这些差异。

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

       首先我们应该清楚的是JDK1.6和JDK1.7中String类的intern方法还是有差别的:

       JDK1.6中的intern:

       调用intern方法的时候首先会去常量池中查看是否存在与当前String值相同的值,如果存在的话,则直接返回常量池中这个String值的引用;如果不存在的话,则会将原先堆中的该字符串拷贝一份到常量池中

       JDK1.7中的intern:

       调用intern方法的时候首先会去常量池中查看是否存在与当前String值相同的值,如果存在的话,则直接返回常量池中这个String值的引用;如果不存在的话,则只会将原先堆中该字符串的引用放置在常量池中,并不会将拷贝整个字符串到常量池中;

       这也就说明,JDK1.6和JDK1.7对于常量池中不存在此字符串的情况处理不同;

       下面通过实例来进行验证和解释:

       实例:

    public static void main(String[] args) {
        String str = "str"+new String("01");①
        str.intern();②
        String str1 = "str01";③
        System.out.println(str == str1);
        
        String str2 = new String("str01");④
        str2.intern();⑤
        String str3 = "str01";⑥
        System.out.println(str2 == str3);
        
        String str4 = "str01";⑦
        String str5 = new String("str")+new String("01");⑧
        str5.intern();⑨
        System.out.println(str4 == str5);
	}
在JDK1.6下输出结果是:

false

false

false

解释:
①执行时会在堆内存创建一个值为"str01"的字符串对象str,同时在常量池创建一个"str"以及"01"常量;

②执行时会首先去常量池中查看是否存在一个值为"str01"的常量,发现不存在,JDK1.6的做法就是将该字符串"str01"在常量池中也生成一份;

③执行时会在常量池中创建一个"str01"对象,发现已经存在,因而不会新建;

第一个输出false的原因是:str指向的是堆内存的"str01",而str1指向的是常量池中的"str01";

④执行时会在堆内存创建一个值为"str01"的字符串对象str2,同时在常量池中创建一个值为"str01"的常量;

⑤执行时会首先去常量池中查看是否存在值为"str01"的常量,发现存在,则直接返回这个常量引用;

⑥执行时会在常量池中创建一个值为"str01"的常量,如果发现已经存在,则不会创建;

第二个输出false的原因是:str2指向的是堆内存的"str01",而str3指向的是常量池中的"str01";

⑦执行时会在常量池创建一个值为"str01"的常量;

执行时会在堆内存创建一个值为"str01"的字符串对象str5,同时在常量池创建一个"str"以及"01"常量;

⑨执行时会去常量池查看是否存在值为"str01"的常量,发现存在则直接返回这个常量引用;

第三个输出false的原因是:str5指向的是堆内存的"str01",而str4指向的是常量池中的"str01";

在JDK1.7下输出结果是:

true

false

false

解释:

发现只有第一个输出结果不一样,所以我们只解释第一个的原因:

执行时会在堆内存创建一个值为"str01"的字符串对象str,同时在常量池创建一个"str"以及"01"常量;(这点和JDK1.6没什么区别)

执行时会首先去常量池中查看是否存在一个值为"str01"的常量,发现不存在,JDK1.7的做法就是将堆内存中"str01"的引用复制到了常量池中

执行时会在常量池中创建一个"str01"对象,发现已经存在,因而不会新建;

那么此时的str和str1都将指向的是堆内存中的"str01"的值,所以两者相等;

### Java 中 `String.intern()` 机制详解 #### 工作原理 `String.intern()` 是一种用于获取字符串常量池中对应字符串的方法。当调用此方法时,如果字符串已经存在于常量池,则返回池中的实例;否则会将当前字符串放入常量池并返回这个新的引用。 对于直接使用构造函数创建的字符串对象如 `new String("abc")` ,它会在堆上分配一个新的对象而不是优先查找常量池中存在的相同内容的对象[^1]。这意味着即使有相同的字符串值,在不同地方通过这种方式创建出来的对象也是不同的实体,它们不会共享同一份内存地址,从而可能导致不必要的重复数据以及增加垃圾收集的压力。 为了优化这种情况下的性能问题和节省内存开销,可以利用 `intern()` 方法让这些具有相等字符序列但是独立构建于堆上的字符串能够指向同一个位于永久代(PermGen space 或 Metaspace)内的唯一表示形式: ```java // 创建两个逻辑上相等但实际上不是同一个对象的字符串变量 String str1 = new String("hello"); String str2 = "hello"; System.out.println(str1 == str2); // 输出 false 表明两者并非同一对象 // 使用 intern 后再做对比 str1 = new String("hello").intern(); System.out.println(str1 == str2); // 此时输出 true 显示二者确实是指向了相同的实例 ``` 上述代码片段展示了如何借助 `intern()` 实现字符串间的高效比较及其带来的好处——减少冗余副本数量的同时提高了程序运行效率[^2]。 #### 使用场景 - **大量重复字符串处理**:在应用程序中有许多完全一致却各自占据单独空间的小型文本串时,采用 `intern()` 可有效降低总体消耗; - **哈希表键名管理**:作为 Map 结构里的 key 值时尤为适用,因为这能确保所有映射关系基于独一无二的标准参照物建立起来而不受外部因素干扰; - **配置文件解析器**:读取 XML/YAML 等格式化文档期间遇到频繁出现的关键字或属性名称亦可考虑应用此类技术手段加以简化维护成本。 需要注意的是虽然 `String.intern()` 能够带来诸多便利之处但也存在一定局限性比如可能引起 PermGen/Metaspace 溢出风险所以在实际开发过程中应当权衡利弊合理选用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值