Hashset的Hashcode和Equals

Lol*_*lly 46 java equals hashcode hashset

请澄清我对Hashset的疑问.考虑以下代码,

class Person
{
    String name;

    Person(String n)
    {
        name=n; 
    }
    public String getName()
    {
        return name;   
    }

    @Override
    public boolean equals(Object arg0) {

        System.out.println("in equals");

        Person obj=(Person)arg0;

        System.out.println("1st "+getName());
        System.out.println("2nd "+obj.getName());

        if(this.getName().equals(obj.getName()))
        {
                return true;
        }
        return false;
    }


    @Override
    public int hashCode() {

        System.out.println("in hash code");
        System.out.println(" value is "+Integer.valueOf(name.charAt(0)));
        return Integer.valueOf(name.charAt(0));
    }
}
Run Code Online (Sandbox Code Playgroud)

在主要我有以下代码

Person obj1=new Person("bcd");

Person obj2=new Person("cde");

Person obj3=new Person("abc");

Person obj4=new Person("abc");
Run Code Online (Sandbox Code Playgroud)

现在,如果我将这些对象添加到hashset

Set<Person> sset=new HashSet<Person>();

sset.add(obj1);
sset.add(obj4);
sset.add(obj2);
sset.add(obj3);
Run Code Online (Sandbox Code Playgroud)

我得到了这个输出

in hash code                                                                      
value is 98    
in hash code   
value is 97    
in hash code    
value is 99    
in hash code    
value is 97  
in equals  
1st abc     
2nd abc
Run Code Online (Sandbox Code Playgroud)

问题1:为什么equals()函数只被调用一次来检查obj3和obj4?为什么没有检查其余的物体?

问题2:如果答案是因为它们都有相同的哈希码,那么只会调用equals,那么为什么不调用下面的代码

sset.add(obj1);
sset.add(obj4);
sset.add(obj2);
sset.add(obj4);
Run Code Online (Sandbox Code Playgroud)

输出是:

in hash code  
value is 98  
in hash code   
value is 97   
in hash code   
value is 99   
in hash code  
value is 97 
Run Code Online (Sandbox Code Playgroud)

即使将两个相同的对象添加到具有相同哈希码的哈希集中,它也不会进入equals()方法.

问题3:我迭代了上面的值并打印了内容,但是没有调用hashcode或equals.什么时候它真的有用来覆盖hashcode和equals方法?

问题4:什么时候会hashCode()equals()被称为?

Eri*_*rik 53

  1. equals如果hashCode不同,则无需拨打电话.
  2. 有没有需要调用hashCode如果(obj1 == obj2).
  3. 没有必要hashCode和/或equals只是迭代 - 你不是在比较对象
  4. 当需要区分对象时.

  • 不需要hashCode和/或等于迭代 - 你不是在比较对象---那么如果它不在迭代中调用hashcode,它将如何获得当前桶? (5认同)
  • `HashMap`实际上计算哈希以找到正确的桶并在将对象标识与`=='或调用`equals()`进行比较之前使用它,但否则这是正确的.在第二个例子中没有调用`equals()`因为`==`已经检测到重复. (3认同)

Zug*_*alt 19

如果您了解集合,特别是HashSets的工作方式,我认为您的问题都将得到解答.集合是唯一对象的集合,Java定义唯一性,因为它不等于任何其他对象(等于返回false).

HashSet利用哈希码来加快速度.它假设两个相等的对象将具有相同的哈希码.但是,它并不假设具有相同哈希码的两个对象意味着它们是相等的.这就是为什么当它检测到冲突的哈希码时,它只会与具有相同哈希码的集合中的其他对象(在您的情况下为一个)进行比较.


小智 11

根据来自javasourcecode.org的jdk源代码,HashSet使用HashMap作为其内部实现,关于HashSet的put方法的代码如下:

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;
    }
Run Code Online (Sandbox Code Playgroud)

规则首先检查哈希,然后检查引用,然后调用equals方法将对象放入.