面试官:说说equals和== 的区别?
01、对于 == 来说:
如果比较的是基本数据类型变量,比较两个变量的值是否相等。(不一定数据类型相同)
如果比较的是引用数据类型变量,比较两个对象的地址值是否相同,即两个引用是否指向同一个对象实体。
测试基本数据类型:
int a = 20;
int a = 20;
double c = 20.0;
char i = 20;
char j = 'A';
boolean boo = true;
System.out.println(c==a);//true
System.out.println(a==i);//true
System.out.println(boo==a);//编译不通过
其中, 布尔类型不能参与运算符==比较,
报以下错误:
Operator ‘==’ cannot be applied to ‘boolean’, ‘int’
测试引用数据类型:
public class Student {
private int age;
private String name;
public Student(int age, String name) {
this.age = age;
this.name = name;
}
public static void main(String[] args) {
Student stu1 = new Student(11, "张三");
Student stu2 = new Student(11,"张三");
System.out.println(stu1==stu2);
}
}
分析:
虽然两个对象的属性值都相等。
但是系统会为每一个new的对象分配内存地址值。(存放在堆中)
两个对象的引用指向不同的地址值,所以结果为false。
我来考考你:
这段代码很熟悉吧,为什么结果是true?
String str = new String("hello");
String str2 = new String("hello");
System.out.println(str.equals(str2));//true
这是因为String类对equals方法进行了重写,源码如下:
public boolean equals(Object anObject) {
//判断调用equals方法的对象和 形参引用地址是否相等
if (this == anObject) {
return true;
}
//判断左边形参引用是否是右边String类的实例对象
if (anObject instanceof String) {
//将形参引用强制转换为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;
}
可能你又有疑问,下面代码为什么是true?
String str = "hello";
String str2 = "hello";
System.out.println(str==str2);//true
分析:
str2创建对象的时候发现常量池中已经存在 “hello”,这时不需要创建新的对象直接将str2引用指向"hello",这样str和str2引用指向同一个地址值,所以结果为true。
知道了以上这一点,总结一下equals的用法。
02、对于 equals 来说:
如果类中重写了equals方法,比较内容是否相等。
String、Date、File、包装类都重写了Object类的equals方法。
如果类中没有重写equals方法,比较地址值手否相等(是否指向同一个对象实体)。
Student stu1 = new Student(11, "张三");
Student stu2 = new Student(11,"张三");
System.out.println(stu1.equals(stu2));//false
既然equals比较的是内容是否相同,为什么结果还是false呢?
回顾知识:
在Java中我们知道任何类的超类都是Object类,Student类也继承Object类。
查看Object类中的equals方法也是 == 比较(也就是比较地址值),因此结果当然是false。
public boolean equals(Object obj) {
return (this == obj);
}
既然这样我们如何保证两个对象内容相同呢?
这里就需要我们去重写equals方法?
@Override
public boolean equals(Object obj){
if (this == obj){
return true;
}
if (obj instanceof Student) {
Student stu = (Student)obj;
return this.age == stu.age && this.name.equals(stu.name);
}
return false;
}
测试:
true
如果对你有帮助,帮我三连 ~~~!