您的位置:首页 > 编程语言 > Java开发

字符串常量池初识

2016-07-20 23:56 218 查看
先科普一下字符串常量池的概念:

Java中字符串对象创建有两种形式,一种为字面量形式,如String str = "abc";,另一种就是使用new这种标准的构造对象的方法,如String str = new String("abc");,这两种方式我们在代码编写时都经常使用,尤其是字面量的方式。然而这两种实现其实存在着一些性能和内存占用的差别。这一切都是源于JVM为了减少字符串对象的重复创建,其维护了一个特殊的内存,这段内存被成为字符串常量池或者字符串字面量池。


工作原理

当代码中出现字面量形式创建字符串对象时,JVM首先会对这个字面量进行检查,如果字符串常量池中存在相同内容的字符串对象的引用,则将这个引用返回,否则新的字符串对象被创建,然后将这个引用放入字符串常量池,并返回该引用。

举例说明

字面量创建形式

String str1 = "abc";


JVM检测这个字面量,这里我们认为没有内容为abc的对象存在。JVM通过字符串常量池查找不到内容为abc的字符串对象存在,那么会创建这个字符串对象,然后将刚创建的对象的引用放入到字符串常量池中,并且将引用返回给变量str1。

如果接下来有这样一段代码

String str2 = "abc";


同样JVM还是要检测这个字面量,JVM通过查找字符串常量池,发现内容为”abc”字符串对象存在,于是将已经存在的字符串对象的引用返回给变量str2。注意这里不会重新创建新的字符串对象。

验证是否为str1和str2是否指向同一对象,我们可以通过这段代码

System.out.println(str1 == str2);


结果为true。

使用new创建

String str3 = new String("abc");


当我们使用了new来构造字符串对象的时候,不管字符串常量池中有没有相同内容的对象的引用,新的字符串对象都会创建。因此我们使用下面代码测试一下,

String str3 = new String("abc");
System.out.println(str1 == str3);


结果如我们所想,为false,表明这两个变量指向的为不同的对象。

intern

对于上面使用new创建的字符串对象,如果想将这个对象的引用加入到字符串常量池,可以使用intern方法。

调用intern后,首先检查字符串常量池中是否有该对象的引用,如果存在,则将这个引用返回给变量,否则将引用加入并返回给变量。

String str4 = str3.intern();
System.out.println(str4 == str1);


输出的结果为true。

下面通过反射写一个demo,让体会更深刻点

package test;

import java.lang.reflect.Field;

public class Test06 {

public static void main(String[] args) throws SecurityException, NoSuchFieldException, IllegalArgumentException, IllegalAccessException {
String str="abc";
Field field=String.class.getDeclaredField("value");
field.setAccessible(true);
char[] arr=(char[])field.get(str);
arr[1]='a';
arr[2]='a';
System.out.println(str);
System.out.println(str=="abc");
System.out.println(str.equals("abc"));
System.out.println("abc");
}
}


输出结果

aaa
true
true
aaa


这里说下出现这结果的原因,第一个输出str,因为java是动态语言,这里通过反射在运行时修改str常量池里的字面值,所以输出为aaa,第二个因为是同属一个常量池字面值,当然地址是一样的,第三个String的equal比较,地址都一样,当然equal就一样了,第四个输出abc变成aaa,更好的说明了str的字面值被修改成了aaa了
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  java jvm string