Java中String的比较

来源:互联网 发布:海岛奇兵狙升级数据 编辑:程序博客网 时间:2024/05/10 22:38
例子A:    
Java代码    
 String str1 = "java"; 
 String str2 = "java"; 
 System.out.print(str1==str2); 
地球上有点Java基础的人都知道会输出false,因为==比较的是引用,equals比较的是内容。不是我忽悠大家,你们可以在自己的机子上运行一 下,结果是true!原因很简单,String对象被放进常量池里了,再次出现“java”字符串的时候,JVM很兴奋地把str2的引用也指向了 “java”对象,它认为自己节省了内存开销。不难理解吧 呵呵  
例子B:  
Java代码  
 String str1 = new String("java"); 
 String str2 = new String("java"); 

 System.out.print(str1==str2);

看过上例的都学聪明了,这次肯定会输出true!很不幸,JVM并没有这么做,结果是false。原因很简单,例子A中那种声明的方式确实是在 String常量池创建“java”对象,但是一旦看到new关键字,JVM会在堆中为String分配空间。两者声明方式貌合神离,这也是我把“如何创 建字符串对象”放到后面来讲的原因。大家要沉住气,还有一个例子。  

例子C:   
Java代码   
 String str1 = "java"; 
 String str2 = "blog"; 
 String s = str1+str2; 

 System.out.print(s=="javablog");

再看这个例子,很多同志不敢妄言是true还是false了吧。爱玩脑筋急转弯的人会说是false吧……恭喜你,你会抢答了!把那个“吧”字去掉你就完 全正确。原因很简单,JVM确实会对型如String str1 = "java"; 的String对象放在字符串常量池里,但是它是在编译时刻那么做的,而String s = str1+str2; 是在运行时刻才能知道(我们当然一眼就看穿了,可是Java必须在运行时才知道的,人脑和电脑的结构不同),也就是说str1+str2是在堆里创建的, s引用当然不可能指向字符串常量池里的对象。没崩溃的人继续看例子D。  

例子D:  

Java代码 

 String s1 = "java"; 

 String s2 = new String("java"); 
 System.out.print(s1.intern()==s2.intern()); 
JDK 1.6 intern():返回字符串对象的规范化表示形式。一个初始为空的字符串池,它由类String 私有地维护。当调用 intern 方法时,如果池已经包含一个等于此String 对象的字符串(用equals(Object) 方法确定),则返回池中的字符串。否则,将此String 对象添加到池中,并返回此String 对象的引用。它遵循以下规则:对于任意两个字符串st,当且仅当 s.equals(t)true 时,s.intern() == t.intern() 才为true。 

例子E:  
Java代码  
   String str1 = "java"; 
   String str2 = new String("java"); 
   System.out.print(str1.equals(str2));//true

无论在常量池还是堆中的对象,用equals()方法比较的就是内容,就这么简单!  

原创粉丝点击