java String的intern方法

时间:2021-10-12 22:06:07

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

 JDK1.6中的intern:  

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

JDK1.7中的intern:  

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

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

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

实例:

java" id="highlighter_247495">
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
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"的值,所以两者相等;

以上就是对JDK1.6和JDK1.7中String类的intern方法的对比,是有差别的,有需要的朋友可以参考下。