面试的时候经常被问到这样一个问题:
String s = new String("abc");//此行执行完成后,内存中有几个String对象??
String s1 = "abc";//此行代码执行完成后内存中有几个String对象?
String s2 = new String("abc");//此行代码执行完成后内存中有几个内存对象??
首先看上面的三行代码和后面的问题,大家都知道第一行代码的问题答案是:2个对象,
那么具体是那两个对象,不知道有没有想过,在JDK1.5 API参考手册是这样说的:
Initializes a newly created String
object so that it represents the same sequence of characters as the argument; in other words, the newly created string is a copy of the argument string. Unless an explicit copy of original
is needed, use of this constructor is unnecessary since Strings are immutable.
注意红色的字体,这句话说明当你使用String s = new String("........")方式创建一个字符串对象的时候,那么系统会在内存中划分出一个叫做String pool的内存块,这个String pool存放的是参数字符串对象,
String s = new String("abc");那么按照JDK说明,系统先创建一个参数字符串对象“abc”放入String pool中,那么这就是第一个String对象。
那么new String("abc")是将String pool中的值为“abc”对象作为参数传递过来,创建在内存的堆中创建一个新的String对象,那么实际上新创建的字符串是该参数字符串的一个副本,一个拷贝,所以会出现两个字符串对象的结果
第二行代码:String s1 = "abc",它执行完成之后,系统内存中有两个字符串对象,为什么呢???
此中方法创建String对象的时候,系统会首先在String pool中查找是否存在一个值为“abc”的字符串,如果存在的
话,那么就将这个值为“abc”对象的引用赋给s1,如果不存在,在String pool中创建一个新的字符串对象,值为
“abc”,因为第一行代码执行完成后,String pool中存在一个值为“abc”的字符串,那么系统会直接将这个字符
串的引用赋给s1,所有第二行代码没有创建一个新的String对象,如果这样写String s1 = “abcd”,那么就会创建
一个新的字符串对象,因为系统在String pool中查找不到这个值为“abcd”的String 对象,那么这个对象创建后也
放在String pool中。
接上面继续:
第三行代码:String s2 = new String("abc")这行代码执行完成之后,系统内存中有三个String对象
执行期间,系统会到String pool中查找有没有值为“abc”的对象,如果的话,那么将String pool中值为“abc”
的对象对象引用作为副本,在内存堆中创建一个值为“abc”的对象(JAVA中的对象创建在堆上),将此对象的引用
赋给s2,如果不存在的话,同样就会是第一行代码的结果