答案是两个。
String s=”123” 创建一个对象“123” 放入常量池中
String s=new String(“123”) 创建两个对象,首先new String()的参数是String对象,创建“123”放入常量池,然后在new一个对象放入堆中。
具体代码分析:
/**
* 字符串(final修饰)是不变的对象,意识是说,字符串内容一但被创建不可改变
* 若想改变某一个字符串的内容一定会创建新对象
* @param args
*/
public static void main(String[] args) {
/*
* java为了提高性能,静态字符串(字面量、常量、常量连接的结果)在常量池中创建
* ,并尽量使用同一个对象,重用静态字符串;
* 对于重复出现的字符串直接量,JVM会首先在常量池中查找,如果存在即返回该对象
*/
String s1="123abc";
String s2="123abc";
System.out.println(s1==s2);
//!此时S1与s2的!!地址!!相等。原因:因为s2的内容与s1相同,在java中为了重用,检查到常量池中已经有相同内容存在,
//就让s2指向堆中的地址与s1的地址相同,都是同一个对象。所以为true
String s3=new String("123abc");
System.out.println(s1==s3);
//!!此时s3是被new出来的,不看常量池,直接是个新对象(新地址),此时两个对象
//这与s1指向的对象不相同,所以此时结果为false
s2+="!";
System.out.println(s1==s2);
//此时s2被改变了,在运行期创建的新的对象,此时不相等。此时有三个对象
String s4="123abc!";
System.out.println(s2==s4);
//由于s2创建的新对象是在运行期,常量池中并没有创建,
//所以在S4未被创建时,常量池中只有"123abc"
//所以s2指向的对象与s4指向的对象不同,结果为false
String s5="123"+"abc";
System.out.println(s1==s5);
/*
* 编译器的优化措施:
* 当一个计算表达式计算符两边都是字面量时,会直接计算结果,然后将结果编译在 class文件中
* 例如现在在字节码文件中变成String S5="123abc";
* 所以s5继续引用常量池中的对象,地址相同,结果为true
*/
String str1="123";
String str2="abc";
String s6=str1+str2;
System.out.println(s1==s6);
/*
* 此时计算符两边都是变量,两个变量拼接直接创建新对象不看常量池
*/
}
代码结果:
true
false
false
false
true
false