String: 由intern看String

String这个类,是Java中很经典的类,从我们学Java的那天起就与他结下“梁子”,在以后的日子他更是如影随形,对你不离不弃。所以,了解String,理解String很有必要。

1. 简介

看看JDK中对String类的描述:

  1. public final class String extends Object  
  2. implements Serializable, Comparable<String>, CharSequence  
 

该类是一个final类,子类无法继承。实现了Serializable、Comparable<String>、CharSequence三个接口。很多公司的Java面试题,都会与String有关,毕竟处理字符串是很频繁的,再者很多算法也和String有关。

2. intern方法

    intern  
    public String intern()  
    返回字符串对象的规范化表示形式。  
    一个初始为空的字符串池,它由类 String 私有地维护。  
    当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(用 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并返回此 String 对象的引用。  
    它遵循以下规则:对于任意两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。  
    所有字面值字符串和字符串赋值常量表达式都使用 intern 方法进行操作。字符串字面值在 Java Language Specification 的 §3.10.5 定义。  
    返回:  
    一个字符串,内容与此字符串相同,但一定取自具有唯一字符串的池。  

 

在具体介绍intern方法之前,举个例子,以便于更好地说明我的问题。
[java] view plaincopyprint?

    package mark.zhang;  
    public class Linux {  
          
        public static void main(String[] args) {  
            String s1 = "ubuntu";  
            String s2 = "ubuntu";  
            System.out.println(s1.equals(s2));//true  
            System.out.println(s1 == s2);//true  
              
            String s3 = new String("ubuntu");  
            String s4 = new String("ubuntu");  
            System.out.println(s3.equals(s4));//true  
            System.out.println(s3 == s4);//false  
        }  
    }  

 


我该保证,大多数人都是知道结果的。不过在这里,我还是想用内存来分析一下结果。

 

字符串常量会放在常量池中,(常量池主要存放字符串常量和基本类型常量(public static final))。有些人的博客中说常量放在data segement中,我想data segement主要用来存放静态变量(这里的变量是指成员变量,因为static不可以修饰局部临时变量)和字符串常量和基本类型常量等。所以,data segement可以分为静态区域以及常量池等。

 

知道这些概念之后,分析上面代码就会显得很轻松啦,来吧,一起分析!!!

  1. String s1 = "ubuntu";  
 

这句代码,执行时会在常量池中查找是否有ubuntu,如果有的话,就指向它,没有的话就自己创建。接着,创建s2,同理s2找到ubuntu,如是就不会自己再在常量池中创建ubuntu啦。这样一来,s1和s2都指向了常量池中的ubuntu。

既然s1与s2都指向一个地方,所以 == 就是true。

  1. String s3 = new String("ubuntu");  
  2. String s4 = new String("ubuntu");  
 

那么,我们看看,这两句代码的内存分析。首先需要明白,new出来的东西放在堆内存中。s3会先在常量池中看看有没有ubuntu,没有的话就先在常量池中创建一个,然后对中对象实例其实是常量池中的一份拷贝。s4发现常量池中已经存在ubuntu,直接在堆中拷贝一份。

从上面来看,s3、s4分别指向不同地址的ubuntu,所以二者就不会==。

好了,回到今天的主题intern方法。我想,如果你明白上面的内容,下面所讲的东西对您来说就会很简单,跟喝水差不多,呵呵!!

  1. package mark.zhang;  
  2. public class Linux {  
  3.       
  4.     public static void main(String[] args) {  
  5.         String s1 = "ubuntu";  
  6.         String s2 = new String("ubuntu");  
  7.         System.out.println("invoke intern before: ");  
  8.         System.out.println(s1 == s2);//false  
  9.         s2 = s2.intern();  
  10.         System.out.println("invoke intern after: ");  
  11.         System.out.println(s1 == s2);//true  
  12.     }  
  13. }  
 

也许,你会问结果怎么会与自己预想不一样??从这个例子,可以明白intern返回的是常量池中的字符串的值。只有这样,两个对象才会==,这也充分说明无论以哪种方式创建的字符串对象,都会先在常量池中创建一个字符串(内容即字面值)。再看下面一个例子,算是对intern及以上所说的一点补充吧?!

  1. package mark.zhang;  
  2. public class Linux {  
  3.       
  4.     public static void main(String[] args) {  
  5.           
  6.         String s1 = new String("redhat");  
  7.         String s2 = new String("redhat");  
  8.         s2 = s2.intern(); // 返回常量池中的redhat  
  9.         // 但是s1指向堆里面的对象,而不是常量池中的redhat  
  10.         System.out.println(s2 == s1);//false  
  11.           
  12.         String s3 = new String("redhat");  
  13.         String s4 = new String("redhat");  
  14.         s3 = s3.intern(); // 返回常量池中的redhat  
  15.         s4 = s4.intern(); // 返回常量池中的redhat  
  16.         System.out.println(s3 == s4);//true   
  17.     }  


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值