最近在复习几个类中String类的问题让我思考再三,现分享出来,欢迎指正
初学之时老师肯定讲过 == 和 equals 的区别,就由这个问题展开思考。一般的时候 用equals比较字符串的值,
String s1="abc";
String s2="abc";
System.out.print(s1.equals(s2)); //true
==用来比较指向的地址(对象)
String s1="abc";
String s2="abc";
System.out.print(s1==s2); //true
其实 equals方法 是String类中自己重写了的,用来比较每个字符是否相同
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String aString = (String)anObject;
if (coder() == aString.coder())
return isLatin1() ? StringLatin1.equals(value, aString.value)
: StringUTF16.equals(value, aString.value);
}
}
return false;
}
关于 new String()创建对象的个数说法不一(这里==比较的是是引用地址) 但是new出来的对象 保存在堆中 是共识 , 常量 在常量池
String s1 = "abc";
String s2 = new String("abc");
System.out.println(s1==s2); // false
String s1 ="sss";
String s2 = "sss";
System.out.println(s1==s2); // true
String s1 ="abc";
String s2 = "def";
String s3="abcdef";
String s4=s1+s2; // 会在堆中生成新对象 但指向常量池中的s3
System.out.println(s3==s4); // false
String s3="abc"+"def"; //常量池
String s4="abcdef"; //常量池
System.out.println(s3==s4); //true
如果字符串在常量池中或者 堆中出现过 那么它(JVM)就会 把引用 返回给引用对象
再反过来看 就感觉很好理解了
String str1 = "hello";
String str2 = "helloworld";
String str3 = str1+"world";//不能确定为常量(会在堆创建String对象)
String str4 = "hello"+"world";//确定为常量,直接到常量池中引用
System.out.println(str2==str3);//fasle
System.out.println(str2==str4);//true
System.out.println(str3==str4);//fasle
new String() 如果是之前未出现的字符串 那么就是创建的两个对象
如果在字符串常量池去寻找没有abc这个字符,就在栈中创建一个abc字符串对象。
然后new String()把这个对象拷贝到堆中,然后返回一个对象的引用,所以产生两个对象
String s1 ="abc";
String s2 = new String("abc");
System.out.println(s1==s2); // false
编辑
观察 0,7
ldc标示从常量池获取一个引用入栈
String s2 = new String("abc");
String s1 ="abc";
System.out.println(s1==s2);
编辑
所以我认为 这两种方式 都会生成 两个对象