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
对象的引用。它遵循以下规则:对于任意两个字符串s
和 t
,当且仅当 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()方法比较的就是内容,就这么简单!