equals方法用于比较两个引用数据类型是否相等,如String或者自己定义的类都属于引用数据类型。“==”则用于比较基本数据类型是否相等。
下边通过一个测试例逐行解析下equals方法原理。
首先直接看下面代码:( equals方法重写部分在67-74)
1 import jdk.swing.interop.SwingInterOpUtils;
2
3 public class Test01 {
4 public static void main(String[] args) {
5 Mytime m1 = new Mytime(2020,12,14);
6 //调用toString(),
7 String t = m1.toString();
8 System.out.println(t);
9 // 输出引用的时候,会自动调用该引用的toString方法。//ctrl + 鼠标左键打开源码
10 System.out.println(m1);
11 //调用equals()
12 Mytime m2 = new Mytime(2020,12,14);
13 // Mytime mytime = (Mytime)m1;
14 boolean f = m1.equals(m2);
15 System.out.println(f);
16 System.out.println(m1.day == m2.day);
17 Test m3 = new Mytime(2020,12,14);
18 System.out.println(m3.getClass());
19 }
20 }
21 class Mytime extends Test{
22 int year;
23 int month;
24 int day;
25 public Mytime(int year, int month, int day) {
26 this.year = year;
27 this.month = month;
28 this.day = day;
29 }
30 public void setYear(int year) {
31 this.year = year;
32 }
33 public void setMoonth(int month) {
34 this.month = month;
35 }
36 public void setDay(int day) {
37 this.day = day;
38 }
39 public int getYear() {
40 return year;
41 }
42 public int getMoonth() {
43 return month;
44 }
45 public int getDay() {
46 return day;
47 }
48 @Override
49 public String toString() {
50 return "Mytime{" +
51 "year=" + year +
52 ", month=" + month +
53 ", day=" + day +
54 '}'+"--"+getClass()+"--"+this.getClass();
55 }
56 /* @Override
57 public boolean equals(Object o) {//这里的Object是基类,所有的类型都能通过这里调用,object o = m2,(或object o =new Mytime()
58 if (this == o) return true;//
59 if (o == null || getClass() != o.getClass()) return false;//o.getClass() =Mytime,
60 Mytime mytime = (Mytime) o;
61 //Mytime mytime = (Mytime) o;
62 return year == mytime.year &&
63 month == mytime.month &&
64 day == mytime.day;
65 }*/
66 @Override
67 public boolean equals(Object o) {
68 if (this == o) return true;
69 if (o == null || getClass() != o.getClass()) return false;
70 Mytime mytime = (Mytime) o;
71 return year == mytime.year &&
72 month == mytime.month &&
73 day == mytime.day;
74 }
75 }
76 class Test{
77
78 }
如上面equals部分(67-74)所示(由IDEA自动生成)。下面是逐行解释:
67行:
在本Test例中:首先以下代码(14行)调用equals方法,
14 boolean f = m1.equals(m2);
在67行中的equals(Object o),可这样理解:Object o = m2或者可以理解成Object o = new Mytime(),因为m2 =new Mytime()。Object为基类。
68行:
this存放了本对象(此处为m1)的内存地址,o存放了m2的内存地址。如果内存地址一样,肯定表示同一个对象;返回true。
69行:
o为null 返回false很好理解;getClass()返回本对象即m1的类名,o.getClass()返回o的类名不是Object,而是m2的类型名,原因在于:Object o = m2 (Object o =new Mytime()),返回的类名就是m2的。
在上面测试例中对此进行了测试,如下:
17 Test m3 = new Mytime(2020,12,14);
18 System.out.println(m3.getClass());
返回的m3.getClass()为Mytime,而不是Test。
m1的类名和m2的类名不同的话肯定不是同一个对象,返回false。
70行:
还是由于67行方法入口的Object o = new Mytime(),
Object类中肯定没有Mytime类中的属性,所以o作为内存地址虽然指向了一个Mytime对象,但是由于Object类中没有相应的属性,就无法通过o.来获取Mytime中的属性。
于是,就需要70行,把Object类型的o强转为Mytime类型的,之后就可以进行下一步的类中属性的比较。注:能执行到这一步,说明m1和m2是同一类了,但由于Object o的存在不能直接获取m2的内部属性。
71~74行就不需要解释了。