`
yuwenlin2008
  • 浏览: 127274 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Object类的equals方法和hashCode方法

阅读更多

Object 类是所有类的父类,Object类中定义了很多重要的方法,有些基础方法是必须要搞清楚的,今天我们就来学习下Object类中的equals方法和hashCode方法。

 

一、equals方法

首先我们来看下Object类的equals方法的源码:

public boolean equals(Object obj) {
        return (this == obj);
}

很明显它是比较两个对象的引用(即内存地址)是否相等。如果你不知道这个,想当然的以为它比较的是内容,比如我们要比较两个用户对象是否相等:

User实体类:

public class User {
	
	private int userId;
	private String userName;
	private int age;
	
	public User(int userId, String userName, int age) {
		this.userId = userId;
		this.userName = userName;
		this.age = age;
	}
	
	public int getUserId() {
		return userId;
	}
	public void setUserId(int userId) {
		this.userId = userId;
	}
	public String getUserName() {
		return userName;
	}
	public void setUserName(String userName) {
		this.userName = userName;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
}

测试代码: 

public class UserTest {
	
	public static void main(String[] args) {
		User u1 = new User(1, "z3", 25);
		User u2 = new User(1, "z3", 25);
		boolean b = u1.equals(u2);
		System.out.println(b);
	}

}

执行结果为:false,为什么呢,因为你调用的是Object类的equals方法,它比较的是两个对象的引用,即内存地址,在Java虚拟机的堆上是两块独立的内存空间,绝对不相等的。并不是我们想要的结果,那应该要怎么做呢?自己重写Object类的equals方法就好了:

我们在User实体类中添加equals方法:

        @Override
	public boolean equals(Object obj) {
		if (obj == null)
			return false;
		if(obj instanceof User) {
			User other = (User) obj;
			if (age != other.age)
				return false;
			if (userId != other.userId)
				return false;
			if (userName == null) {
				if (other.userName != null)
					return false;
			} else if (!userName.equals(other.userName))
				return false;
		}
		return true;
	}

再次执行测试代码,结果为:true

我们可以看到重写的equals方法,比较了User类的每一个成员变量,那成员变量之间怎么比较呢,如果是基本数据类型如int,float,double,boolean等,直接比较其值就行了。

如果是字符串,直接调用字符串的equals方法即可,因为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类的eqauls方法比较的是每一个char字符。其实查看Jdk源码,可以发现,各基本数据类型的包装类如Integer,Float,Double,Boolean等都重写了Object类的equals方法。因为实际业务需要我们去比较对象或变量的内容而不是引用。

 

二、hashCode方法

Object类的hashCode方法源码如下:

public native int hashCode();

它是一个本地方法,返回一个int类型的整数。 

Java为每一个对象提供一个int类型的hashCode,其目的是快速查找定位对象,猜测啊,在底层有一张Hash表,存储对象的hashCode和内存地址的映射,这样方便JVM在内存中能够快速查找定位到某个对象,不然,内存中那么多的对象,JVM如何查找一个对象呢。

 

hashCode有几个重要特性:

1.hashCode是为了查找对象或元素提高性能

2.如果两个对象equals相等,那么两个对象的hashCode一定相等

3.如果两个对象的hashCode相等,两个对象的equals不一定相等

4.如果要重写对象的equals方法,尽量要重写对象的hashCode方法

 

接下来,我们一一说明:

第1点,hashCode是为了查找对象或元素提高性能,这在上面已经说了JVM要查找内存中的对象可以快速定位。还有就是集合类的实现,比如Set集合,元素不允许重复,如果集合中现在已经有1000个元素,那么第1001个元素加入集合时,它就要调用1000次equals方法。这显然会大大降低效率。于是采用Java的哈希表原理,先根据元素的hashCode定位元素位置,如果该位置元素不存在,则将元素插入集合的该位置,如果已经存在,则equals比较其内容,如果内容一致则元素重复了,否则采用链式数据结构存储到该位置(HashMap的实现)。

 

第2点,如果两个对象equals相等,那么两个对象的hashCode一定相等。这个上面的Set集合就已经解释了,如果两个对象equals相等,hashCode不相等,那么根据Set集合实现会将两个元素存储到不同的位置,那么这就违背Set集合元素不允许重复了。

 

第3点,如果两个对象的hashCode相等,两个对象的equals不一定相等。这个我们看下Set的实现HashSet,其底层是HashMap的实现:

public V put(K key, V value) {
    if (key == null)
        return putForNullKey(value);
    int hash = hash(key.hashCode());
    int i = indexFor(hash, table.length);
    for (Entry<K,V> e = table[i]; e != null; e = e.next) {
       Object k;
        if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
            V oldValue = e.value;
            e.value = value;
            e.recordAccess(this);
            return oldValue;
        }
    }

    modCount++;
    addEntry(hash, key, value, i);
    return null;
}

根据HashMap的put方法实现,我们可以看到首先根据key的hashCode计算出元素在哈希表的位置,如果该位置上已经有元素(说明这两元素的hashCode相等),再equals比较两元素的内容,如果不相等,则以链表的方式将两元素存储在该位置,这充分说明了两元素hashCode相等,但equals不一定相等。

 

第4点,如果要重写对象的equals方法,尽量要重写对象的hashCode方法,这个其实就是保证第2点的实现,如果不重写hashCode方法,还是以Set集合为例,就无法保证元素不重复。可以查看jdk源码,像String类,Integer类等只要重写了equals方法的,都重写了hashCode方法。

 

 

总结,我们在开发中,养成良好的习惯,每写一个实体类,就重写它的equals方法和hashCode方法,要么都不写,要写两个都要写,可以减少不必要的问题发生。

 

 

 

<audio controls="controls" style="display: none;"></audio>

0
0
分享到:
评论

相关推荐

    关于Object中equals方法和hashCode方法判断的分析

    在 Java 中,Object 类提供了两个重要的方法:equals 方法和 hashCode 方法。这两个方法都是用于比较两个对象是否相等的,但它们的实现机理和作用域却有所不同。 equals 方法是用于比较两个对象是否相同的。它的...

    重写equals和hashcode方法_equals_重写equals和hashcode方法_

    在Java编程语言中,`equals()` 和 `hashCode()` 方法是Object类中的两个核心方法,所有类都默认继承自Object类。这两个方法在处理对象比较和集合操作时起着至关重要的作用。当我们创建自定义类并需要对对象进行精确...

    hashcode和equals方法

    equals()和hashcode()这两个方法都是从object类中继承过来的。当String 、Math、还有Integer、Double。。。。等这些封装类在使用equals()方法时,已经覆盖了object类的equals()方法.

    equals与hashCode方法讲解

    equals 方法和 hashCode 方法是 Java 语言中两个重要的方法,它们都是在 Object 类中定义的。equals 方法用于比较两个对象是否相等,而 hashCode 方法用于返回对象的哈希码。 在 Java 的 Object 类中,equals 方法...

    Java理论与实践:hashCode()和equals()方法

    本文介绍了Java语言不直接支持关联数组,可以使用任何对象作为一个索引的数组,但在根Object类中使用 hashCode()方法明确表示期望广泛使用HashMap。理想情况下基于散列的容器提供有效插入和有效检索;直接在对象模式...

    Java equals 方法与hashcode 方法的深入解析.rar

    在Java编程语言中,`equals()`方法和`hashCode()`方法是两个非常重要的概念,它们主要用于对象的比较和哈希表的高效运作。本解析将深入探讨这两个方法的用途、实现原理以及它们之间的关联。 首先,`equals()`方法是...

    Java中equals,hashcode和==的区别

    "Java中equals、hashcode和==的区别" Java 中 equals、hashcode 和==的区别是 Java 编程语言中一个经常遇到的问题。这三个概念都是用来比较对象的,但是它们之间存在着本质的区别。 首先,==号是Java中的一个...

    equals,hashcode,toString

    在这个类中,很可能已经重写了`equals()`, `hashCode()` 和 `toString()` 方法,以便更好地处理对象的比较、哈希存储和输出信息。 为了实现`DBObject`类的正确行为,开发者可能考虑了以下几点: 1. 在`equals()`...

    学习Object类——为什么要重写equeals和hashcode方法

    Object 类的 equals 和 hashCode 方法的重要性与实现 在 Java 编程语言中,Object 类是所有类的父类,但是在实际开发中,我们往往需要重写 Object 中的 equals 和 hashCode 方法,以便正确地比较对象的逻辑内容,而...

    Java的Object类讲解案例代码 equals()、hashCode()、finalize()、clone()、wait()

    Object类是所有Java类的根类,它定义了一些常用的方法,例如equals()、hashCode()、toString()等。本案例代码将详细展示Object类的使用方法,并提供一些实际场景下的案例,以帮助开发者更好地理解和运用这些方法。 ...

    java中hashcode()和equals()方法详解

    在Java编程语言中,`hashCode()`和`equals()`方法是非常重要的概念,它们不仅对于深入理解Java内存管理至关重要,也是实现自定义类的关键部分之一。本文将详细介绍这两个方法的工作原理、使用场景以及它们之间的关系...

    Java_重写equals()和hashCode()

    首先,`equals()` 方法是Object类中的一个基础方法,用于比较两个对象是否相等。默认情况下,它比较的是对象的内存地址,也就是引用是否相同。但在实际开发中,我们往往需要根据对象的内容来判断它们是否相等,这时...

    set接口经常用的hashCode和equals方法详解

    ### set接口中hashCode和equals方法详解 #### 一、引言 在Java编程语言中,`Set`接口作为集合框架的重要组成部分,在实现无重复元素的数据结构方面扮演着关键角色。为了确保元素的唯一性,`Set`接口依赖于对象的`...

    关于hashCode()和equals()的本质区别和联系

    本文将详细介绍 hashCode() 和 equals() 的本质区别和联系,并探讨在创建 Java 类时如何定义这些方法。 hashCode() 方法 hashCode() 方法是 Object 类中的一个方法,它返回对象的哈希码值。哈希码值是一个整数,它...

    equals与hashCode在实际开发中的重写写法

    在Java编程语言中,`equals()` 和 `hashCode()` 方法是两个非常重要的成员,尤其是在处理对象比较和集合操作时。这两个方法通常与`Object`类中的默认实现相关联,但为了在实际开发中实现正确的对象比较和哈希表操作...

    Java容器集合(equals 和 hashCode+基础数据结构+ArrayList+Vector和LinkedList)

    其中,equals和hashCode方法是Java容器集合中两个非常重要的方法,本文将详细介绍这两个方法,并结合ArrayList、Vector和LinkedList三个常见的容器集合。 一、equals方法 equals方法是Java中用于比较两个对象是否...

    关于重写equals,hashcode以及compareTo方法!

    equals()方法是Object类中的一个方法,它用于比较两个对象是否相等。然而,它的默认实现是比较对象的引用(地址),而不是比较对象的实际内容。因此,在某些情况下,我们需要重写equals()方法,使其比较对象的实际...

    Java中的equals和hashCode方法详解1

    在Java编程语言中,`equals()`和`hashCode()`方法是对象的基本组成部分,它们主要用于对象的比较和存储。这两个方法在`java.lang.Object`类中定义,因此所有的Java类都默认继承了这两个方法。然而,根据具体的应用...

    Java中hashCode和equals方法的正确使用

    在这篇文章中,我将告诉大家我对hashCode和equals方法的理解。我将讨论他们的默认实现,以及如何正确的重写他们。我也将使用Apache Commons提供的工具包做一个实现。  hashCode()和equals()定义在Object类中,这...

Global site tag (gtag.js) - Google Analytics