一: new String("hello") 和 String a = "hello" 的区别
public static void method(){
String a = "hello";
String b = "hello";
String c = "h"+"ello";
String d = new String("hello")
System.out.prinlnt(a==b); //true
System.out.prinlnt(a==c); //true
System.out.prinlnt(a==d); //false
}
-
1.前者跟new一个对象是一样的,后者是常量池,但也是会创建对象;
-
2.在JVM中,为了减少对字符串变量的重复创建,其拥有一段特殊的内存空间,这段内存被称为
字符串常量池(constant pool)
或者是字符串字面量池
。 -
3.使用 String a = "Hello" 创建了一个对象,此时默认字符串常量池中没有内容为“Hello”的对象存在。当JVM在字符串常量池中没有找到内容为"Hello"的对象时,就会创建一个内容为"Hello"的对象,并将它的引用返回给变量a。
-
4.再使用 String b = "Hello" 的时候
JVM会在字符串常量池中查询是否有内容为“Hello”的对象。因为此时字符串常量池中已经有内容为“Hello”的对象了,故JVM不会创建新的地址空间,而是将原有的“Hello”对象的引用返回给b -
5.所以此时变量a和变量b拥有的是同一个对象引用,即指向的是同一块内存地址,因此使用
==
操作符对a和b进行比较,结果为true。
int n=3;
int m=3;
System.out.println(n==m); //true int是基本数据类型,比较它的值,==就是用来比较值是否相等
String str = new String("hello");
String str1 = new String("hello");
String str2 = new String("hello");
System.out.println(str1==str2); //false 他们比较的引用的地址
str1 = str;
str2 = str;
System.out.println(str1==str2); // 把str堆中的地址复制给str1 和 str2 ,所以他们相等
- 1.基本数据就是比较的值
- 2.对于非基本数据类型的变量,上面的str1就是引用类型的变量,引用类型的变量存储的并不是 “值”本身,而是于其关联的对象在内存中的地址
String str1 = new String("hello");
String str2 = new String("hello");
System.out.println(str1==str2); //false
System.out.println(str1.equals(str2)); //true ,原因是String重写了equals方法,比较字符串的内容是不是一样的
- 1.String str1,这句话声明了一个引用类型的变量,此时它并没有和任何对象关联。
- 2.str1= new String("hello"),通过new String("hello")来产生一个对象(也称作为类String的一个实例),并将这个对象和str1进行绑定:
- 3.那么str1指向了一个对象,此时变量str1中存储的是它指向的对象在内存中的存储地址,并不是直接存储的字符串"hello"。
- 4.因此在用==对str1和str2进行第一次比较时,得到的结果是false。因此它们分别指向的是不同的对象,也就是说它们实际存储的内存地址不同。
- 5.而在第二次比较时,都让str1和str2指向了str指向的对象,那么得到的结果毫无疑问是true。
二.equals比较
equals方法是基类Object中的方法,因此对于所有的继承于Object的类都会有该方法。
这是基类的equals方法的实现
public boolean equals(Object obj){
return (this == obj);
}
在Object类中,equals方法是用来比较两个对象的引用是否相等,即是否指向同一个对象。
public static void main(String[] args) {
String str1 = new String("hello");
String str2 = new String("hello");
System.out.println(str1.equals(str2)); //true
Person one = new Person("yang",19);
Person two = new Person("yang",19);
System.out.println(one.equals(two)); //false
}
为什么下面一段代码的输出结果是true?
String类中equals方法的具体实现
image.png
- String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。
- 其他的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。
-
为什么两个看起来相同的Person对象,为什么是false,因为Person的对象中没有重写equals,在没有重写的情况下,还是调用基类的equals,也就是比较他们的内存地址值,所有要重写equals,这样就会输出true
image.png
三.总结
1)对于==,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;如果作用于引用类型的变量,则比较的是所指向的对象的地址
2)对于equals方法,注意:equals方法不能作用于基本数据类型的变量
- 如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;
- 诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。
网友评论