这是 java编程思想 第五章---初始化与清理 的一题
练习2 :
创建一个类,它 包含了一个定义时就被初始化了的String域,以及另一个通过构造器初始化的String域,这两种方式有何差异.
看这一段程序以及结果, 首先明确三个知识点:
1.String是引用对象
2.equals() 比较的是值 //有误:对于字符串满足,但是对于其他引用对象,equals与==一样,均是比较地址
3. "=="是比较的 地址
String s1 = "abc"; String s2 = "abc"; String s3 = new String("abc"); String s4 = new String("abc"); System.out.println(s1==s2); //true System.out.println(s3==s4); //false System.out.println(s1==s3); //false System.out.println(s1.equals(s2)); //true System.out.println(s1.equals(s3)); //true
java有两类存储类型,一类是基本类型的变量,如int,float等, 一类是引用对象, 如String, 自己创建的各种class, 数组等等, 就是需要new的
数据一般存储在两个地方, 栈中存放基本类型和对象的引用变量, new出来的对象放在堆中
除此之外, 还有块区域常量池, String s1 = "abc" , "abc"就存在常量池中, 在我们又创建一个String s2 = "abc"后, java底层在常量池中找到了"abc", 就让s2指向这个值, 不再重新创建. 所以第一个为true,
而我们new的则在 堆中, 因为常量池是存储已知的, 而new是需要编译后才知道的
再看一个例子
String s1 = "aaa"; String s2 = "bbb"; String s3 = "aaabbb"; System.out.println(s3 == s1 + s2); //false System.out.println(s3 == ("aaa" + "bbb")); //true
System.out.println(s3 == "aaa" + new String("bbb"); //false
这个例子的中 s3 == s1 + s2 当进行比较时, java 为 "s1 + s2" 在堆中开辟一个内存, 然后再拼接字符串
"aaa" + "bbb" 则进行拼接后, 在常量池发现"aaabbb" 所以相等.
"aaa" + new String("bbb") 因为含有new, 则是运行是才可以确定结果, 会在堆中, 不相等
不懂常量池的可以看看这篇文章: https://www.jianshu.com/p/55f65dac1b4b