先看一个示例程序
String s1 = new StringBuilder().append("ja").append("va1").toString();
String ss1 = s1.intern();
System.out.println(ss1 == s1);
String s2 = "java1";
String ss2 = s2.intern();
System.out.println(ss1 == ss2);
System.out.println(s2 == ss2);
System.out.println(s1 == s2);
//输出结果:
true
true
true
true
再来一个示例程序
String s1 = new StringBuilder().append("ja").append("va1").toString();
// System.out.println(s1.intern() == s1);
String s2 = "java1";
System.out.println(s2 == s1.intern());
System.out.println(s1 == s2);
//输出结果:
true
false
String.intern()方法注释:
A pool of strings, initially empty, is maintained privately by the class String
When the intern method is invoked, if the pool already contains a
string equal to this String object as determined by
the equals(Object) method, then the string from the pool is
returned. Otherwise, this String object is added to the
pool and a reference to this String object is returned.
翻译过来就是:一个初始为空的字符串池,它由类String独自维护。当调用 intern方法时,如果常量池已经包含一个等于此String对象的字符串(用equals(oject)方法确定),则返回池中的字符串。否则,将此String对象添加到池中,并返回此String对象的引用。
示例程序1中调用intern方法返回的是s1对象在堆内存中的地址,再次定义变量s2,其值仍然是s1引用,此处为了防止两次intern方法返回的不同特意进行了比较,证明是相同的。
示例程序2中,第一次比较为true是因为s2变量导致常量池中已经有java1了,所以调用intern方法返回的是池中字符串,而不是引用,所以s1和s2比较就是false。
所以就如intern方法注释,intern可能返回字符串也可能返回引用。
最后再补两段程序:
String s1 = new String("java1");
System.out.println(s1.intern() == s1);
String s2 = "java1";
System.out.println(s1.intern() == s2);
输出结果:
false
true
String s3 = new StringBuilder().append("j").append("av").toString();
System.out.println(s3.intern() == s3);
String s4 = new StringBuilder().append("j").append("av").toString();
System.out.println(s3 == s4);
System.out.println(s4.intern() == s3);
System.out.println(s3.intern() == s4.intern());
输出结果:
true
false
true
true
注意:此处不能用java字符串测试,猜测是默认内置了,结果将是false,false,false,true
jdk6和7的可以参考美团技术团队的文章:# 深入解析String#intern