String s=new String("abc");//JVM现在String pool中找有没有“abc”的字符串,没有就创建一个该字符串对象“abc”,该对象在String pool中并不是在堆中,这就是很多教程上所说的垃圾对象,然后JVM会通过String pool中对象在堆里也创建一个“abc”String对象,句柄s指向的是堆中的对象;
System.out.println(s==s.intern());//JDK文档中有关intern()函数的解释:
返回字符串对象的规范化表示形式。
一个初始为空的字符串池,它由类 String 私有地维护。
当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(用 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并返回此 String 对象的引用。
它遵循以下规则:对于任意两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。
显然s.intern()指向的是String pool中的"abc"而非堆中的"abc",结果当然是false,此时必有s.equal(s.intern())==true;
String s1="abc";//JVM会去Stiring pool中找是否有“abc”的字符串存在,如果存在,则,s1指向它,当然是存在的了,因此才会有 System.out.println(s.intern()==s1);返回true
String s2=new String("abc");//使用new创建的肯定是不同的对象,虽然String pool中已经有“abc”对象存在;同理,s2指向的是堆中新创建的“abc”,s2和s指向的堆中的不同对象,但是内容相同“abc”。
因此有如下结果:
System.out.println(s==s2);//返回false
System.out.println(s.intern()==s2.intern());//返回true
另外一个小测试:
String hello="hello";
String hel="hel";
String lo="lo";
System.out.println(hello=="hel"+"lo");//true,连接符两边都是常量的则从池中找并返回。
System.out.println(hello=="hel"+lo);//false,连接符两边有变量则在堆中重新生成了一个对象。
System.out.println(s==s.intern());//JDK文档中有关intern()函数的解释:
返回字符串对象的规范化表示形式。
一个初始为空的字符串池,它由类 String 私有地维护。
当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(用 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并返回此 String 对象的引用。
它遵循以下规则:对于任意两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。
显然s.intern()指向的是String pool中的"abc"而非堆中的"abc",结果当然是false,此时必有s.equal(s.intern())==true;
String s1="abc";//JVM会去Stiring pool中找是否有“abc”的字符串存在,如果存在,则,s1指向它,当然是存在的了,因此才会有 System.out.println(s.intern()==s1);返回true
String s2=new String("abc");//使用new创建的肯定是不同的对象,虽然String pool中已经有“abc”对象存在;同理,s2指向的是堆中新创建的“abc”,s2和s指向的堆中的不同对象,但是内容相同“abc”。
因此有如下结果:
System.out.println(s==s2);//返回false
System.out.println(s.intern()==s2.intern());//返回true
另外一个小测试:
String hello="hello";
String hel="hel";
String lo="lo";
System.out.println(hello=="hel"+"lo");//true,连接符两边都是常量的则从池中找并返回。
System.out.println(hello=="hel"+lo);//false,连接符两边有变量则在堆中重新生成了一个对象。