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
equals如果hashCode不同,则无需拨打电话.hashCode如果(obj1 == obj2).hashCode和/或equals只是迭代 - 你不是在比较对象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方法将对象放入.
| 归档时间: |
|
| 查看次数: |
94020 次 |
| 最近记录: |