前言:面试被问到答了个七七八八,但是总是不太熟练于是写下了这一篇博客,期待自己将他牢记于心。
在学习Java前期你是否碰到过这样的疑问?
String str1=new String("aaa");
String str2=new String("aaa");
System.out.println("第一次比较:"+(str1==str2));
//
System.out.println("第一次比较进行equals比较===》"+ (str1.equals(str2)));
上面的在比较什么为什么==和equals得出来的结果不一样呢?
咱们分开来进行讲述。
1、关系操作符“==”
说的简单一点,==就是比较值是否相同,有一个前提就是进行比较的对象是基本数据类型
。说到这里你有没有在自己脑子里面回想那8种基本数据类行呢?🤭 那么接下来我们一起来复习Java最基本的东西,不能好高骛远哦。
接下来还有三种引用类型。接口,类,数组
。
示例:
int i=2;
int m=2;
System.out.println(i==m);//基本数据类型比较的是值。
String st=new String("qqq");
String str1=new String("aaa");
String str2=new String("aaa");
System.out.println("第一次比较:"+(str1==str2));
System.out.println("第一次比较进行equals比较===》"+ (str1.equals(str2)));
str1=st;
str2=st;
System.out.println("交换之后进行比较"+(str1==str2));
这样你会发现第三次输出和第四次的输出是一样的。因为str1
和str2
都引用了同一个对象,所以你会发现==如果用于引用类型的话比较的是对象的内存地址。
2、方法equals源码刨析
equals
方法原本是Object类中定义的,因为Object是所有类的父类,所以其他的类也可以重写equals
方法,(你有没有忘记equals本身是那个类里面的呢?🤭)
equals本身比较的是对象的地址。
Object源码如下:
public boolean equals(Object obj) {
return (this == obj);//比较内存地址
}
因为上面介绍过==比较引用类型比较的是内存地址,所以说equals方法本身比较的是内存地址,但是因为很多类重写了父类的equals方法,把equals方法进行了改造,所以说,有点equals方法比较的是对象的值。
String源码如下:
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
/* 重写父类方法添加判断条件,比较其值 */
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
你可以看到相比较而言比父类中多了一个判断,使用数组判断每个Char是否相等,也就是判断其值是否相等,所以说如果内存地址不一样,但值是一样的,也可以返回true
。
所以说不要傻乎乎的只回答equals比较的是值。把原理给他说出来会是很棒的点😄。但是有一点equals不能判断基本的数据类型
。
接下来我们来看一下其他类重写的equals方法源码。
Integer类中equals方法
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
该方法单纯判断其值是否相等。其他基本数据类型的源码类似就不一一展示了。
Date时间类的源码:
public boolean equals(Object obj) {
return obj instanceof Date && getTime() == ((Date) obj).getTime();
}
...
public long getTime() {
return getTimeImpl();
}
private final long getTimeImpl() {
if (cdate != null && !cdate.isNormalized()) {
normalize();
}
return fastTime;
}
可以看出比较的也是基本数据类型,原理就是他们把一个引用类型变成一个基本数据类型进行比较
。
3、总结
- 对于
==
,如果作用于基本数据类型的变量,则直接比较其存储的 “值”是否相等。- 如果作用于引用类型的变量,则比较的是所指向的对象的地址。
- 对于equals方法,注意:equals方法不能作用于基本数据类型的变量。
- 如果没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;
- 如果重写后根据具体重写的逻辑判断,来判断是比较其值,还是他们的内存地址。(一般重写后都是比较值)
结束: 通过这一篇博客我想我已经牢牢的掌握了他们的区别,那么身为读者的你们呢?如果以上有何问题欢迎提出指正,也可在留言区交流一下你们面试的囧事😊。