Java中的字符常量池
Java中字符串对象创建有两种形式,一种为字面量形式,如String str = "wuyang";
另一种就是使用new这种标准的构造对象的方法,如String str = new String("wuyang");
这两种方式我们在代码编写时都经常使用,尤其是字面量的方式。然而这两种实现其实存在着一些性能和内存占用的差别。这一切都是源于JVM为了减少字符串对象的重复创建,其维护了一个特殊的内存,这段内存被成为字符串常量池或者字符串字面量池。
工作原理
当代码中出现字面量形式创建字符串对象时,JVM首先会对这个字面量进行检查,如果字符串常量池中存在相同内容的字符串对象的引用,则将这个引用返回,否则新的字符串对象被创建,然后将这个引用放入字符串常量池,并返回该引用。
举例
字面量创建形式:String str1 = "wuyang";
JVM检测这个字面量,这里我们认为没有内容为wuyang的对象存在。JVM通过字符串常量池查找不到内容为wuyang的字符串对象存在,那么会创建这个字符串对象,然后将刚创建的对象的引用放入到字符串常量池中,并且将引用返回给变量str1。
再创建个str2:String str2 = "wuyang";
同样JVM还是要检测这个字面量,JVM通过查找字符串常量池,发现内容为”wuyang”字符串对象存在,于是将已经存在的字符串对象的引用返回给变量str2。注意这里不会重新创建新的字符串对象。
检验下是否为同一对象:System.out.println(str1 == str2);
结果为:true
使用new创建:String str3 = new String("wuyang");
当我们使用了new来构造字符串对象的时候,不管字符串常量池中有没有相同内容的对象的引用,新的字符串对象都会创建。
测试下:
String str3 = new String("wuyang");
System.out.println(str1 == str3);
结果为:fasle
,表明这两个变量指向的为不同的对象。
intern
对于上面使用new创建的字符串对象,如果想将这个对象的引用加入到字符串常量池,可以使用intern方法。
调用intern后,首先检查字符串常量池中是否有该对象的引用,如果存在,则将这个引用返回给变量,否则将引用加入并返回给变量。
String str4 = str3.intern();
System.out.println(str4 == str1);
结果为:true
更多详情见: http://droidyue.com/blog/2014/12/21/string-literal-pool-in-java/