String str1="i"与 String str2=new String("i") 是不一样的。
String str1="i" 中 str 指向的对象是 常量池中的 “i”
String str2=new String("i") 中 str 指向的是 堆中的 “i”
理解这一点需要我们了解常量池,堆和栈的区别,
常量池存放字符串常量和基本类型常量(public static final)。
栈:栈内存首先是一片内存区域,存储的都是局部变量,凡是定义在方法中的都是局部变量(方法外的是全局变量),,变量有自己的作用域,一旦离开作用域,变量就会被释放。栈内存的更新速度很快,因为局部变量的生命周期都很短。
堆:存储的是数组和对象(其实数组就是对象),凡是new建立的都是在堆中,堆中存放的都是实体(对象),如果一个指向实体的变量消失了,这个实体不会消失,还可以用,所以堆是不会随时释放的,但是栈不一样,栈里存放的都是单个变量,变量被释放了,那就没有了。堆里的实体虽然不会被释放,但是会被当成垃圾被Java垃圾回收机制回收
当String str1="i" 时,JVM首先在常量池中寻找是否存在 “i” 这个对象,如果不存在,则在常量池中创建一个"i" 对象,并返回给引用对象:我们在栈里创建的 str1 变量。如果存在就直接将地址返回给引用
当String str2=new String("i") 时,会先去常量池中查找是否已经有了 “i” 对象,如果没有则在常量池中创建一个此字符串对象,然后在堆中再创建一个常量池中此 “i” 对象的拷贝对象(如果常量池中有,就直接在堆中创建拷贝对象)。将这个堆中产生的对象返回给引用 str2。
有这样一道面试题:String s = new String(“xyz”);产生几个对象?
答:一个或两个,如果常量池中原来有”xyz”,就是一个,没有就是两个。