1 说出字符串比较后的结果
package string;
public class Demo2_String {
public static void main(String[] args) {
String st1 = "abc"; //常量池中创建字符串’abc‘
String st2 = "abc"; //发现常量池中已经存在对象’abc‘ 直接复用
System.out.println(st1 == st2); //
System.out.println(st1.equals(st2));
}
}
输出结果分析: true, true
先看第一个打印语句,在Java中==这个符号是比较运算符,它可以基本数据类型和引用数据类型是否相等,如果是基本数据类型,==比较的是值是否相等,如果是引用数据类型,==比较的是两个对象的内存地址是否相等。字符串不属于8中基本数据类型,字符串对象属于引用数据类型,在上面把“abc”同时赋值给了st1和st2两个字符串对象,指向的都是同一个地址,所以第一个打印语句中的==比较输出结果是 true
然后我们看第二个打印语句中的equals的比较,我们知道,equals是Object这个父类的方法,在String类中重写了这个equals方法,在JDK API 1.6文档中找到String类下的equals方法,点击进去可以看大这么一句话“将此字符串与指定的对象比较。当且仅当该参数不为null,并且是与此对象表示相同字符序列的 String 对象时,结果才为 true。” 注意这个相同字符序列,在后面介绍的比较两个数组,列表,字典是否相等,都是这个逻辑去写代码实现。由于st1和st2的值都是“abc”,两者指向同一个对象,当前字符序列相同,所以第二行打印结果也为true。
2 下面这句话在内存中创建了几个对象
String st1 = new String("abc");
分析:内存图
当我们看到了new这个关键字,就要想到,new出来的对象都是存储在堆内存。然后我们来解释堆中对象为什么是常量池的对象的拷贝副本。“abc”属于字符串,字符串属于常量,所以应该在常量池中创建,所以第一个创建的对象就是在常量池里的“abc”。第二个对象在堆内存为啥是一个拷贝的副本呢,这个就需要在JDK API 1.6找到String(String original)这个构造方法的注释:初始化一个新创建的 String 对象,使其表示一个与参数相同的字符序列;换句话说,新创建的字符串是该参数字符串的副本。所以,答案就出来了,两个对象
通过上面内存图判断下面字符串比较结果
demo 1
package string;
public class Demo2_String {
public static void main(String[] args) {
String st1 = new String("abc"); //创建两个对象’abc‘,分别在堆中和常量池中
String st2 = "abc"; //复用常量池中的对象’abc‘
System.out.println(st1 == st2); //==比较值和地址,st1和st2地址不同,一个在常量池中,一个在堆中,故返回 fasle
System.out.println(st1.equals(st2)); //比较两个字符串的值,值都是abc,故返回true
}
}
demo2
package string;
public class Demo2_String {
public static void main(String[] args) {
String st1 = "a" + "b" + "c"; //常量拼接时,jvm会自动优化,把拼接后的结果放到常量池中,故此语句执行后,常量池中存在字符串’abc‘
String st2 = "abc"; //常量池中已经存在,复用即可
System.out.println(st1 == st2); //都在常量池中,地址和值都相等,结果为true
System.out.println(st1.equals(st2)); //都在常量池中,值相等,结果为true
}
}
demo3
package string;
public class Demo2_String {
public static void main(String[] args) {
String st1 = "ab"; //在常量池中创建字符串对象’ab‘
String st2 = "abc"; //在常量池中创建字符串对象’abc‘
String st3 = st1 + "c"; //在堆中创建独享st3,引用对象字符串在拼接的过程中,地层使用的是StringBuffer,拼接过程中会创建新对象,所以st3对象存放在堆中
System.out.println(st2 == st3); //值相等,地址不相等,故结果为false
System.out.println(st2.equals(st3)); //值相等,故结果为true
}
}