深入理解equals与==
前言:
这几天读者在学习JDK的源码,在研究Object类的时候发现了平常经常遇到的一个问题。所以,针对equals与==做一次总结,方便以后自己重新回来学习。
-
技术点:
-
1、继承
可以基于已经存在的类构造一个新类(子类/导出类),子类可以复用这些类的方法和域,同时子类可以添加新的方法和域,从而实现java的多态。
-
2、重载(overload)
在一个类中,允许相同名字的方法出现多次,但是参数必须不同。ps:这里说明一点点小知识,在java中药重载一个方法,除了要与原方法具有相同的简单名称之外,还要求必须拥有一个与原方法不同的特征签名,但是方法返回值不在这个特征签名中,所以重载仅仅返回值不同是无法通过编译的。以下是错误例子:
package com.brickworkers.entend;
public class Demo {
public String method(String key){
return "返回为String";
}
public int method(String key){
return "返回为int";
}
//是无法通过编译的,因为这两段方法仅仅是返回值不同
}
-
:
-
3、重写(override)
子类沿袭父类的方法,并重新定义它,但是要保证方法的方法名,参数,返回类型与父类一致。
-
4、Object类是所有类的父类
如果问笔者为什么Object是所有类的父类,笔者也打不上来,但是大家应该都清楚每个类都会有一个equals方法,其实这个方法是继承了Object的方法
误区:
-
1、equals与==的比较
在各种面试或者学习过程中,被告知的只是简单的:==是比较内存地址是否相等,equals是比较值是否相等。
解释:
1、我们先看下JDK源码中Object是如何实现equals方法的:
public boolean equals(Object obj) {
return (this == obj);
}
从源码中我们可以看到,其实在最底层equals与==一样都是比较内存地址的。
2、那么为什么在实际中大家都说equals是比较值是否相等的呢?这就是体现了java继承之后方法重写的概念了。前面说过Object是所有类的父类,那么子类就可以重写equals方法(父类中private方法不可重写),我们看下String中的重写equals方法:
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;
}
在String的equals中除了内存地址相同返回为true还存在值的比较,类似的还有int,double等等。
3、那我自己的类也需要重写equals方法该如何写呢?
以下是User类,用于比较:
package com.brickworkers.entend;
public class User {
private String name;
private Integer age;
public User(Integer age){
this.age = age;
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj instanceof User) {
User anotherUser = (User)obj;
if (this.getAge() == anotherUser.getAge()) {
return true;
}
}
return false;
}
public User(String name){
this.name = name;
}
public User(String name, Integer age){
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Integer getAge() {
return age;
}
public void setAge(Integer age) {
this.age = age;
}
@Override
public String toString() {
return "User [name=" + name + ", age=" + age + "]";
}
}
以下为测试demo:
package com.brickworkers.entend;
public class Demo{
public static void main(String[] args) {
User user1 = new User(25);
User user2 = new User(25);
System.out.println(user1.equals(user2));
}
//输出为true
}