前言
发现JavaGuide中对于这里的描述不是特别清楚,重新记录一下
一、 “ ==”符号
如果比较的是基本数据类型变量,比较两个变量的值是否相等。(不一定数据类型相同)
如果比较的是引用数据类型变量,比较两个对象的地址值是否相同,即两个引用是否指向同一个地址值
测试基本数据类型:
int a = 20;
double c = 20.0;
char i = 20;
char j = 'A';
char m = 65;
boolean boo = true;
System.out.println(c==a);//true
System.out.println(a==i);//true
System.out.println(j==m);//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的用法。
二、“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;
}
如果更加全面的重写equals方法,我们可以自动生成一下。(前提你要了解为什么这样)
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Student student = (Student) o;
return age == student.age &&
Objects.equals(name, student.name);
}
三、总结
“==”:
- 如果比较的是基本数据类型变量,比较两个变量的值是否相等。(不一定数据类型相同)
- 如果比较的是引用数据类型变量,比较两个对象的地址值是否相同,即两个引用是否指向同一个地址值。
“equals”:
- 如果类中重写了equals方法,则比较内容是否相等,String、Date、File、包装类都重写了Object类的equals方法。
请注意,例子中展示了Student类的equals方法是继承Object类的,Object类中的equals方法也是 == 比较(也就是比较地址值),如果是这样情况我们需要自己重写equals方法。
- 如果类中没有重写equals方法,比较地址值是否相等(是否指向同一个地址值)。