1、String s = new String(“hello”) 和 String s = “hello”的区别?
字符串作为一个常量是存放在常量池里面的,新建字符串的时候会先从缓冲区里面找有没有这个字符串,如果有直接返回引用,没有的话就在里面创建一个。
String s = “hello”可能创建一个对象或者不创建对象,主要看“hello”在常量池中是否存在。
String s = new String(“hello”)至少创建一个对象,也可能创建两个。因为在使用new关键字的时候,首先会在堆内存里面创建一个String对象,然后这个对象指向常量池中已存在的“hello”或者新创建的“hello”。
2、如何理解String一旦赋值就不能被改变?
String一旦赋值就不能被改变指的是在方法区中的String字符串常量,一旦生成之后就不会再被改变。
要改变String对象,只能在常量池里面重新生成一个新的字符串然后将String对象指向该新生成的字符串。
例如:
String s = "123";
s += "456";
首先,会在常量池中创建一个字符串“123”;
然后,在常量池中创建一个字符串“456”;
最后,再在常量池中创建一个字符串“123456”,s指向“123456”。
3、为什么equals可以比较字符串?
比较两个字符串使用equals()而不用“==”,这是因为String对象重写了equals方法,在equals方法中比较的是两个字符串的内容,而不是引用的地址。
4、字符串相加,是先拼接再分配空间还是先分配空间再相加?
String s1 = "123";
String s2 = "456";
String s3 = "123456";
System.out.println(s3 == s1 + s2);
System.out.println(s3 == "123" + "456");
结果为:
false
true
对生成的.class文件进行反编译,
String s1 = "123";
String s2 = "456";
String s3 = "123456";
System.out.println(s3 == (new StringBuilder(String.valueOf(s1))).append(s2).toString());
System.out.println(s3 == "123456");
s1+s2 是先开辟的空间,然后再进行拼接。而”123” + “456”则在编译时就进行了相加操作,由于在声明s3时常量池中已经有了字符串“123456”,因此在执行”123” + “456”不会再创建新的对象,而是直接从常量池中取出来。这样s3 和 “123456”的地址值就一致了。
字符串如果是变量相加,先开空间,在拼接。
字符串如果是常量相加,是先加,然后在常量池找,如果有就直接返回,否则,就创建。