==,equals(),hashCode()三者的区别和联系

==的作用

对于基本数据类型,比较两个值是否相等
对于引用数据类型,比较两个引用所指向的内存地址是否相等
程序无法控制比较的结果,比较的结果完全由内存地址决定

equals()的作用

==的程序可控制版本,比较的结果可以通过对象equals()方法中的代码决定。
如果不重写对象的equals()方法,==和equals()结果等价。
可以根据业务逻辑重写equals()方法,例如学生对象,业务逻辑中两个学生对象的班级,名字,性别三个相等就判断为同一个对象,但是==判断为不相等。

hashCode()的作用

hashCode()将当前对象作为hash函数的输入参数,输出int值,称为散列码(hashcode)

hashcode的作用:确定对象在hashtable中所在的桶(bucket)的位置(index)。
hash函数:无限个输入参数只对应有限个输出散列码。这样就有可能出现两个不一样的输入对应同一个输出。
如果两个不同对象的hashCode()返回的hashcode相同,那么称这两个对象在这个hash函数上hash冲突。
hash函数的目的:通过CPU运算提高查询的速度

==和equals()的关系?

两个对象==判断为true时,equals()判断一定为true;
两个对象equals()判断为true时,==判断不一定为true。
此时分两种情况:
1. 如果对象对象没有重写equals()方法,equals()判断的结果等价于==的判断结果
2. 如果对象重写equals()方法,除非两个对象是同一个对象,否则判断结果一定为false

equals()和hashCode()关系?

因使用场景而不同,
1. 如果对象不使用在类似HashMap,HashSet等Hash性质的数据结构中,equals()和hashCode()没有任何关系。
2. 如果使用在Hash性质的数据结构中,为了利用Hash的特性,必须保证:两个对象equals()相等,hashCode()必须相等,这个也是java语言规范里面规定的,所以重写对象equals()方法时一定要重写hashCode()方法。
如果两个equals()相等,但是hashCode()不相等,当两个对象都加入到HashSet去重时,两个相等的对象会被存放在不同的桶中,不能达到去重的目的。

重写equals()时如何重写hashCode()方法

  1. 参照《Effective Java》
public class User {
    private String name;
    private int age;
    private String passport;

    //getters and setters, constructor

    @Override
    public boolean equals(Object o) {

        if (o == this) return true;
        if (!(o instanceof User)) {
            return false;
        }

        User user = (User) o;

        return user.name.equals(name) &&
                user.age == age &&
                user.passport.equals(passport);
    }

    //Idea from effective Java : Item 9
    @Override
    public int hashCode() {
        int result = 17;
        result = 31 * result + name.hashCode();
        result = 31 * result + age;
        result = 31 * result + passport.hashCode();
        return result;
    }

}

2.使用Java 7以上版本提供的Objects类生产equals和hashcode

import java.util.Objects;

public class User {
    private String name;
    private int age;
    private String passport;

    //getters and setters, constructor

    @Override
    public boolean equals(Object o) {

        if (o == this) return true;
        if (!(o instanceof User)) {
            return false;
        }
        User user = (User) o;
        return age == user.age &&
                Objects.equals(name, user.name) &&
                Objects.equals(passport, user.passport);
    }

    @Override
    public int hashCode() {
        return Objects.hash(name, age, passport);
    }

}

3.使用Apache Commons Lang提供的工具类

import org.apache.commons.lang3.builder;

public class User {
    private String name;
    private int age;
    private String passport;

    //getters and setters, constructor

     @Override
    public boolean equals(Object o) {

        if (o == this) return true;
        if (!(o instanceof User)) {
            return false;
        }

        User user = (User) o;

        return new EqualsBuilder()
                .append(age, user.age)
                .append(name, user.name)
                .append(passport, user.passport)
                .isEquals();
    }

    @Override
    public int hashCode() {
        return new HashCodeBuilder(17, 37)
                .append(name)
                .append(age)
                .append(passport)
                .toHashCode();
    }

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值