JVM中的字符串常量池
一.字符串常量池的简介
由于String是java开发中最常使用的引用类型之一,它和其他引用类型一样,在创建的时候是非常消耗空间的,于是为了程序提高性能以及减少内存开销,java设计者在JVM层面设计了字符串常量池
二.字符串常量池的优缺点
1.优点
当我们创建一个字符串时,会首先去字符串常量池查看该字符串的值是否存在,如果存在则池中不再创建,如果不存在,那么池中会创建一个该值的字符串,这样就可以避免重复创建相同值的字符串对象,减少内存开销
举个例子:String s=“123”,首先查看字符串常量池中是否有值为"123"的字符串对象,如果有则不再创建,直接该对象把地址值赋值给s,如果没有,那么字符串常量池中创建一个值为"123"的字符串对象,然后把该对象地址值赋值给s
2.缺点
上述说到,当创建一个新的String对象时,会去遍历一遍字符串常量池看是否有该值,这样会消耗一部分时间,降低程序性能,但总的来说,时间开销较小.
三.字符串常量池的设计前提
String类型对象被实例化后,是不可变的,因为如果多个字符串实例对象的值一样,指向字符串常量池中的同一个地址,当一个实例的值改变时,其他实例也要跟着改变,这样就会造成数据冲突.
四.String对象的两种创建方式
1.直接赋值方式,String s=“XXX”
当采用直接赋值的方式时,例如,String s=“abc”,JVM首先回去查看字符串常量池中有没有值为"abc"的String对象,如果有直接把该对象的地址值赋值给s,如果没有就在字符串常量池创建一个值为"abc"的String对象,再把该对象的地址值赋值给s
图解
2.new一个字符串对象
当new一个字符串时,String s = new String(“123”),JVM首先会在堆中创建一个值为"123"的String对象,然后再去字符串常量池中查看是否有"123"的String对象存在,如果有就不创建,如果没有创建.总的来说就是比方法一多了一个在堆中创建一个String对象
五.案例分析
public class TansferValue {
public static void changeValue(String temp){
temp="abc";
return ;
}
public static void main(String[] args) {
String s="123";
changeValue(s);
System.out.println(s);
}
}
解析:String是引用类型,为什么值没有改变呢?,这就是字符串常量池的原因,当创建字符串String s=“123"的时候,字符串常量池中没有该值的对象就创建了一个值为"123"的对象,当调用changeValue()方法时,temp=“abc”,字符串常量池发现没有"abc"这个对象,于是又创建了一个值为"abc"的对象,所以s的值根本没有改变,而是字符串常量池中创建了两个String对象,值分别为"123”,“abc”,这也证明了String的不可改变