Equals和==还分不清?

1. ==

对于基本类型和引用类型 == 的作用效果是不同的,两者比较如下

  • 基本类型:比较的是值是否相同;
  • 引用类型:比较的是引用是否相同;

我们可以来看一个简单的实例

public static void main(String[] args) {
    String x = "Hello";
    String y = "Hello";
    String z = new String("Hello");

    System.out.println(x==y);   // true
    System.out.println(x==z); // false
}

System.out.println(x==y); 对于引用对象的x和y,指向的是同一个字符串,两个引用是相同的,因此输出true。

System.out.println(x==z); 对于字符串z会指向新new出来的一个字符串对象,x与z指向的不是同一个字符串对象。

2. equals

equals 本质上就是 ==,只不过 String 和 Integer 等重写了 equals 方法,把它变成了值比较。看下面的代码就明白了,如果没有重写equals方法,默认是按照==进行比较,如果是普通的对象,则比较的是引用类型。

  • 如果是基本类型,则采用的是Object的equals方法进行比较
  • 如果是字符串,String类重写了equals方法,采用String的equals方法比较
  • 如果是一般对象,并且没有重写equals方法,则默认采用的==进行比较
public class EqualsAndSame {

    public static void main(String[] args) {
        String x = "Hello";
        String y = "Hello";
        Integer a = 1;
        Integer b = 1;
        // 因为比较的是基本类型,采用的基本数据类型重写的equals方法,比较的是值
        System.out.println(a.equals(b));
        // 因为比较的是字符串,采用的String重写的equals方法,比较的是各个字符串
        System.out.println(x.equals(y)); // true

        Person lisi0 = new Person("lisi",21);
        Person lisi1 = new Person("lisi",21);
        // 因为比较的是对象,因为Person类没有重写自己的equals方法,
        // 因此默认采用Object的equals()方法
        System.out.println(lisi0.equals(lisi1));
    }
}


class Person{

    private String name;
    private Integer age;

    public Person(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;
    }
}

1. Object类中的equals()

equals()如果不被重写,则底层比较方式和==基本一致,底层就是通过==来比较的。

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

2. 基本类型的equals比较

/**
 * 本质上比较Integer对象的值
 * @param obj      比较的对象
 * @return
 */
public boolean equals(Object obj) {
    if (obj instanceof Integer) {
        return value == ((Integer)obj).intValue();
    }
    return false;
}

3. String的equals()方法

/**
 * 比较两个字符串是否相等
 * 当anObject对象不为空且该对象与比较对象完全相等时返回true
 * @param anObject
 * @return
 */
public boolean equals(Object anObject) {
    // 1. 如果被比较对象与anObject完全相等时则返回true
    if (this == anObject) {
        return true;
    }

    // 判断字符是否是字符串类型
    /**
     * 2.1 首先判断两个字符串长度是否相等,相等则继续比较内部元素;否则直接返回false
     * 2.2 利用while循环依次比较两个数组内部字符是否相等
     */
    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;
}

修炼到这种功程度,下次面对这种面试题目,可以这么说:


==对于不同类型比较的是不同的,基础类型对比的是值是否相同,引用类型对比的是引用是否相同;而 equals 对于不同的类型比较的是不同的,对于基础类型因此基础类本身实现了equals()方法,其底层比较的是值是否相同,对于String比较的是字符串内容是否完全相同,而对于其他的类型,如果本身没有重写equals()方法,其比较的是引用是否相同,如果实现了equals()方法,则按照实现方法的逻辑进行比较。

第二讲

上面说到了重写equals()方法必须重写hashCode()方法,只是说到了面试中应该怎么回答,如果对于上面回答不是很理解,下面将深入的讲解一番。

1. 源码一窥

hashCode()方法

本身是一个本地方法,也就是说本身是在本地层面进行求解hashCode()的值的,返回值时int。

public native int hashCode();

源码中对于hashCode()也有一些约束,源码备注中的翻译如下:

  1. 如果对象在使用equals方法中进行比较的参数没有修改,那么多次调用一个对象的hashCode()方法返回的哈希值应该是相同的

  2. 如果两个对象通过equals方法比较是相等的,那么要求这两个对象的hashCode方法返回的值也应该是相等的

  3. 如果两个对象通过equals方法比较是不同的,那么也不要求这两个对象的hashCode方法返回的值是不相同的。(换句话说两个对象equals()不相同,这两个对象的hashCode()方法可等可不等)但是我们应该知道对于不同对象产生不同的哈希值对于哈希表(HashMap)能够提高性能。

hashCode()的源码中一直介绍hashMap,那我们可以看看hashMap中对于hashCode()的应用,先看一下HashMap的基本结构(关于hashMap的知识网上有很多知识,此处不再赘述)


那么是如何确定一个数据存储在数组中的哪个位置呢?就是通过hashCode方法进行计算出存储在哪个位置,上面hashCode()规则时说了有可能两个不同对象的hashCode方法返回的值相同,那么此时就会产生冲突,产生冲突的话就会调用equals方法进行比对,如果不同,那么就将其加入链表,如果相同就替换原数据。

本身hashMap容器重写了HashCode()方法的,但是本质上也是调用Object的hashCode()方法

如果自定义对象作为key但是不重写hashCode()方法会怎么样呢?

 

先看一下没有重写hashCode()的对象
Person定义如下:


class Person{

    private String name;
    private Integer age;
    public Person(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;
    }
}

public class EqualsAndSame {

    public static void main(String[] args) {
        Map<Person, Integer> personMap = new HashMap<>();
        Person person1 = new Person("zhangsan",12);
        Person person3 = new Person("zhangsan",12);
        personMap.put(person3, 2222);
        System.out.println(personMap.get(person3));
    }
}

输出结果出人意料确实null,没有按照预获取到2222,为啥会这样呢?

HashMap是采用hashCode()用来计算该对象放入数组中的哪个位置,两个都是重写new的对象,但是由于没有重写,默认都是采用ObjecthashCode(),虽然里面的内容都相同,但是对象所处的地址却不同,所以使用默认的hashCode也就不同,get()方法就以为是获取另一个key值。返回的自然是null

如何解决这个问题呢?
可以重写equals()方法和hashCode()方法,可以通过Idea快捷键自动生成equals()hashCode()方法

class Person{
    ...
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Person person = (Person) o;
        return Objects.equals(name, person.name) &&
                Objects.equals(age, person.age);
    }

    /**
     * 重写的hashCode()方法是通过对象的全部字段来计算hashCode值
     * 最底层是通过Arrays的hashCode()来计算
     * @return
     */
    @Override
    public int hashCode() {
        return Objects.hash(name, age);
    }
}

Objects.hash()底层是通过Arrays.hashCode()实现的

public static int hashCode(Object a[]) {
    if (a == null)
        return 0;
    int result = 1;
    // 遍历对象的每个字段,通过字段的hashCode()方法累加得到
    for (Object element : a)
        result = 31 * result + (element == null ? 0 : element.hashCode());

    return result;
}

重写了之后,两个对象只要各个字段的值相同,对象的hashCode值必定相同,这也就是为什么重写equals()方法的时候要求必须重写hashCode()方法了

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值