一道老生常谈的字符串对象创建问题

    科技2022-07-11  111

    问题和结论

    老生常谈的一个问题,String s = new String(“abc”)创建了几个对象,String s = "abc"又是创建了几个对象?

    先说结论:

    String s = new String(“abc”)创建了1个或2个对象,String s = "abc"创建了1个或0个对象

    接下来我们探究一下String s = new String(“abc”) 和String s = "abc"创建过程的区别。

    String s = new String(“abc”)的创建过程

    系统先在字符串常量池里面寻找是否有一个"abc"的字符串,如果有的话,则在堆中复制一个该字符串,并且将堆中的引用赋给s,这个时候系统只创建了一个对象,即堆中的对象;如果没有的话,则会先在字符串常量池中先创建一个字符串为"abc"的常量,然后再复制到堆里面,最后将堆所在的地址赋给s,这个时候创建了两个对象;

    String s = "abc"的创建过程

    系统先在字符串中寻找是否存在"abc"的常量如果存在,则直接将该"abc"在常量池中的地址赋给s,这个时候,系统没有创建新对象。如果不存在,则在常量池中新建一个"abc"并放入常量池里面,然后再返回该地址,这个时候,系统创建了一个对象。

    示例

    String s = "abc"; System.out.println(s == "abc"); // true

    因为s 和 “abc” 都是指常量池里面"abc" 的地址,所以true;

    String s = new String("abc"); System.out.println(s == "abc"); // false

    因为s 指向的是堆里面新建的对象的地址,而"abc"指向的是常量池里面的地址,所以不等。

    String s = new String("abc"); String s1 = new String("abc"); System.out.println(s == s1); // false

    s和s1都是在堆中新建了不同的对象,虽然内容一样,但是两则是位于堆中不同地址的空间,所以是不相等的。

    String s = "a" + "b"; System.out.println(s == "ab"); // true

    此处虚拟机会做优化,会在常量池里面寻找"a" + “b” 结果后的字符串即"ab",所以两者都是对应常量池中"ab"的地址。

    String s1 = "a"; String s2 = "b"; String str = s1 + s2; System.out.println(str == "ab"); // false

    虽然s1和s2各自指的是常量池里面"a","b"的引用,但是string在做加法或者subString、replace等方法的时候,实际上返回的是new String()的结果,因此str指向的是堆中的地址,所以不相等。

    String s = "abc"; String ss = new String("abc").intern(); System.out.println(s == ss); // true

    虽然此处ss是new出来的新对象,但是由于调用了intern方法,这个方法会返回常量池中相等值的字符串的地址,所以最后ss指向的是常量池中"abc"的地址,所以相等。

    Processed: 0.009, SQL: 8