7

私はいくつかのhashCode+equals + Mapのものをいじっていて、何か...奇妙なものを見つけました。

スニペットは次のとおりです。

class Obj {
    String n;
    Obj(String n) {this.n = n;}

    public int hashCode() {return 0;}
    public boolean equals(Object o) {return false;} // no instance of this class 
                                                    // equals any other instance

}

それから私はこのようなことをしました:

    java.util.Map<Obj,String> map = new java.util.HashMap<Obj,String>();
    Obj o1 = new Obj("1");
    Obj o11 = new Obj("1");
    Obj o2 = new Obj("2");

    map.put(o1,"val 1");
    map.put(o11,"val 2");
    map.put(o2,"val 3");

    p("size = " + map.size()); // obviously 3
    p(map.get(new Obj("1"))); // obviously null     
    p(map.get(o1)); // ...

最後の行は奇妙な部分です。最後の行はを返しますval 1。どうして?equalsメソッドは常にを返しますfalse。これは、==以前に演算子が使用されていたためequalsですか?

洞察をありがとう。

4

2 に答える 2

8

HashMap.javaでは、メソッドは次のとおりです。get

public V get(Object key) {
    if (key == null)
        return getForNullKey();
    int hash = hash(key.hashCode());
    for (Entry<K,V> e = table[indexFor(hash, table.length)];
         e != null;
         e = e.next) {
        Object k;
        if (e.hash == hash && ((k = e.key) == key || key.equals(k)))
            return e.value;
    }
    return null;
}

この行は、を呼び出す前にif (e.hash == hash && ((k = e.key) == key || key.equals(k)))使用しているキーを実際に比較します。平等を排除する試みが失敗するのはそのためです。==equals

于 2012-11-22T23:18:33.583 に答える
2

ハッシュマップの実装は、キーの==と等しいことの両方をチェックします。(hashCodeが同じ場合)

http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/6-b14/java/util/HashMap.java#HashMap.get%28java.lang.Object%29

于 2012-11-22T23:18:13.443 に答える