Java intern() 方法
Java String类
intern() 方法返回字符串对象的规范化表示形式。
它遵循以下规则:对于任意两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。
语法
public String intern ()
参数
返回值
一个字符串,内容与此字符串相同,但一定取自具有唯一字符串的池。
实例
public class Test {
public static void main ( String args []) {
String Str1 = new String ( "www.runoob.com" );
String Str2 = new String ( "WWW.RUNOOB.COM" );
System . out . print ( "规范表示:" );
System . out . println ( Str1 . intern ());
System . out . print ( "规范表示:" );
System . out . println ( Str2 . intern ());
}
}
以上程序执行结果为:
规范表示: www . runoob . com
规范表示: WWW . RUNOOB . COM
Java String类
尽管在输出中调用intern方法并没有什么效果,但是实际上后台这个方法会做一系列的动作和操作。在调用"ab".intern()方法的时候会返回"ab",但是这个方法会首先检查字符串池中是否有"ab"这个字符串,如果存在则返回这个字符串的引用,否则就将这个字符串添加到字符串池中,然会返回这个字符串的引用。
可以看下面一个范例:
String str1 = "a" ;
String str2 = "b" ;
String str3 = "ab" ;
String str4 = str1 + str2 ;
String str5 = new String ( "ab" );
System . out . println ( str5 . equals ( str3 ));
System . out . println ( str5 == str3 );
System . out . println ( str5 . intern () == str3 );
System . out . println ( str5 . intern () == str4 );
得到的结果:
true
false
true
false
为什么会得到这样的一个结果呢?我们一步一步的分析。
第一、str5.equals(str3)这个结果为true,不用太多的解释,因为字符串的值的内容相同。 第二、str5 == str3对比的是引用的地址是否相同,由于str5采用new String方式定义的,所以地址引用一定不相等。所以结果为false。 第三、当str5调用intern的时候,会检查字符串池中是否含有该字符串。由于之前定义的str3已经进入字符串池中,所以会得到相同的引用。 第四,当str4 = str1 + str2后,str4的值也为"ab",但是为什么这个结果会是false呢?先看下面代码:
String a = new String ( "ab" );
String b = new String ( "ab" );
String c = "ab" ;
String d = "a" + "b" ;
String e = "b" ;
String f = "a" + e ;
System . out . println ( b . intern () == a );
System . out . println ( b . intern () == c );
System . out . println ( b . intern () == d );
System . out . println ( b . intern () == f );
System . out . println ( b . intern () == a . intern ());
运行结果:
false
true
true
false
true
由运行结果可以看出来,b.intern() == a和b.intern() == c可知,采用new 创建的字符串对象不进入字符串池,并且通过b.intern() == d和b.intern() == f可知,字符串相加的时候,都是静态字符串的结果会添加到字符串池,如果其中含有变量(如f中的e)则不会进入字符串池中。但是字符串一旦进入字符串池中,就会先查找池中有无此对象。如果有此对象,则让对象引用指向此对象。如果无此对象,则先创建此对象,再让对象引用指向此对象。
当研究到这个地方的时候,突然想起来经常遇到的一个比较经典的Java问题,就是对比equal和==的区别,当时记得老师只是说"=="判断的是"地址",但是并没说清楚什么时候会有地址相等的情况。现在看来,在定义变量的时候赋值,如果赋值的是静态的字符串,就会执行进入字符串池的操作,如果池中含有该字符串,则返回引用。
执行下面的代码:
String a = "abc" ;
String b = "abc" ;
String c = "a" + "b" + "c" ;
String d = "a" + "bc" ;
String e = "ab" + "c" ;
System . out . println ( a == b );
System . out . println ( a == c );
System . out . println ( a == d );
System . out . println ( a == e );
System . out . println ( c == d );
System . out . println ( c == e );
运行的结果:
true
true
true
true
true
true