java中hashcode()和equals()

equals 方法在非空对象引用上实现相等关系:

  • 自反性:对于任何非空引用值 xx.equals(x) 都应返回 true
  • 对称性:对于任何非空引用值 xy,当且仅当 y.equals(x) 返回 true 时,x.equals(y) 才应返回 true
  • 传递性:对于任何非空引用值 xyz,如果 x.equals(y) 返回 true,并且 y.equals(z) 返回 true,那么 x.equals(z) 应返回 true
  • 一致性:对于任何非空引用值 xy,多次调用 x.equals(y) 始终返回 true 或始终返回 false,前提是对象上 equals 比较中所用的信息没有被修改。
  • 对于任何非空引用值 xx.equals(null) 都应返回 false

Object 类的 equals 方法实现对象上差别可能性最大的相等关系;即,对于任何非空引用值 xy,当且仅当 xy 引用同一个对象时,此方法才返回 truex == y 具有值 true)。

注意:当此方法被重写时,通常有必要重写 hashCode 方法,以维护 hashCode 方法的常规协定,该协定声明相等对象必须具有相等的哈希码。


hashCode 的常规协定是:

  • 在 Java 应用程序执行期间,在对同一对象多次调用 hashCode 方法时,必须一致地返回相同的整数,前提是将对象进行 equals 比较时所用的信息没有被修改。从某一应用程序的一次执行到同一应用程序的另一次执行,该整数无需保持一致。
  • 如果根据 equals(Object) 方法,两个对象是相等的,那么对这两个对象中的每个对象调用 hashCode 方法都必须生成相同的整数结果。
  • 如果根据 equals(java.lang.Object) 方法,两个对象不相等,那么对这两个对象中的任一对象上调用 hashCode 方法 要求一定生成不同的整数结果。但是,程序员应该意识到,为不相等的对象生成不同整数结果可以提高哈希表的性能。

实际上,由 Object 类定义的 hashCode 方法确实会针对不同的对象返回不同的整数。(这一般是通过将该对象的内部地址转换成一个整数来实现的,但是 JavaTM 编程语言不需要这种实现技巧。)

1、当使用HashSet时,hashCode()方法就会得到调用,判断存储在集合中的对象是hash code值是否与增加的对象的hash code 值一致。

如果不一致,直接加进去;

如果一致,再进行equals方法比较,equals方法如果返回true,表示对象已经加进去了,就不会再增加新的对象,否则加进去

2、如果我们重写equals方法,那么也要重写hashCode方法,反之亦然。


HashSet存放普通的字符串

package com.bjsxt.str;

import java.util.HashSet;

public class SetTest1 {
	public static void main(String[] args) {
		HashSet set = new HashSet();
		System.out.println(set.add("a"));
		set.add("b");
		set.add("c");
		set.add("d");
		set.add("a");
		System.out.println(set);
	}

}
/**
 * 运行结果: true [d, b, c, a]
 */


HashSet存放没有重写hashcode和equals方法的对象

package com.bjsxt.str;

import java.util.HashSet;

import org.junit.Test;

public class SetTest2 {

	@Test
	public void test1() {
		HashSet set = new HashSet();
		set.add(new People("zhangsan"));
		set.add(new People("lisi"));
		set.add(new People("zhangsan"));
		System.out.println(set);/* 结果:三个对象 */

	}

	@Test
	public void test2() {
		HashSet set = new HashSet();
		People p1 = new People("zhangsan");
		set.add(p1);
		set.add(p1);
		System.out.println(set);/* 结果:[com.bjsxt.str.People@211b37] */
	}

	@Test
	public void test3() {
		HashSet set = new HashSet();
		String s1 = new String("a");
		String s2 = new String("a");
		set.add(s1);
		set.add(s2);
		System.out.println(set);/* 结果:[a] */
		System.out.println(s1.hashCode() == s2.hashCode());// true
	}

}

class People {
	String name;

	public People(String name) {
		this.name = name;
	}
}


HashSet存放重写hashcode和equals方法的对象

package com.bjsxt.str;

import java.util.HashSet;

import org.junit.Test;

public class SetTest3 {
	@Test
	public void test1() {
		HashSet set = new HashSet();
		Student s1 = new Student("zhangsan");
		Student s2 = new Student("zhangsan");
		set.add(s1);
		set.add(s2);
		// 开发中一个认为内容相同的就是同一个对象,不考虑地址
		System.out.println(set);/* 结果:[com.bjsxt.str.Student@aa9c3074] 认为是一个对象 */
		System.out.println(s1.hashCode() == s2.hashCode());// true
	}

}

/** Eclipse可以自动生成equals和hashcode方法 */
class Student {
	String name;

	public Student(String name) {
		this.name = name;
	}

	@Override
	public int hashCode() {
		// 返回name的hashcode来表示当前类的hashcode
		return this.name.hashCode();
	}

	@Override
	public boolean equals(Object obj) {
		if (this == obj) {
			return true;
		}
		// 如果obj不为null,且obj是当前类的实例
		if (null != obj && obj instanceof Student) {
			Student s = (Student) obj;
			if (name.equals(s.name)) {
				return true;
			}
		}
		return false;
	}
}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

时间辜负了谁

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值