经常面试会被问到这两个的区别,比如String s = new String("abc")创建了几个对象,String s = "abc"又是创建了几个对象
ps: String s = new String("abc")创建了1个或2个对象,String s = "abc"创建了一个或0个对象
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"的地址,所以相等。
网友评论