我认为我的问题很简单,但我找不到解决方案,所以我决定在这里问一下。我需要的是使用这样的自定义键类型HashMap
:
HashMap<Pair<Integer, Integer>, StrategyPoint> myMap = new HashMap<Pair<Integer, Integer>, StrategyPoint> ();
但是我在这里遗漏了一些东西,因为HashMap
停止正常工作。首先,Key变得不唯一,并且keySet
中可以找到具有相同值的Pair的不同实例。包含键功能也不像我想的那样工作:)。
我明显错过了一些东西,我更应该以某种方式定义一种方法来比较我的Pair
类中的实例。但是我尝试在我的compareTo
课程中使用Pair
实现Comparable,但它仍然无效。有什么建议吗?
我的原始代码有点混乱,读起来不友好,所以我举了一个例子来说明我的问题。
以下是代码:
HashMap<Pair<Integer, Integer>, StrategyPoint> myMap = new HashMap<Pair<Integer, Integer>, StrategyPoint> ();
Pair<Integer, Integer> myPair = new Pair<Integer, Integer>(2,2);
StrategyPoint myPoint= new StrategyPoint(2, 2, 5, 5, false);
myMap.put(myPair, myPoint);
Pair<Integer, Integer> searcher = new Pair<Integer, Integer> (0,0);
searcher.setFirst(2);
searcher.setSecond(2);
System.out.println(myMap.containsKey(searcher));
System.out.println(myMap.containsKey(myPair));
执行的结果是:
false
true
我已经调试了它并正在正确填充搜索器实例,但似乎HashMap
拒绝在其keySet
中找到它。
答案 0 :(得分:14)
您必须在equals
课程上正确实施hashCode
和Pair
。
HashMap
使用这些方法来区分和散列密钥类。
答案 1 :(得分:5)
您需要覆盖班级equals
中的Pair
。此方法的实现定义了Pair
的两个对象如何被视为相等。
每当您覆盖equals
时,您必须始终覆盖hashcode
。
以下是覆盖equals
而不是hashcode
时出错的问题(来自Effective Java,Second Ed。):
根据类的不同,两个不同的实例在逻辑上可能相等 equals方法,但对于Object的hashCode方法,它们只是两个 没有太多共同点的物体。因此Object的hashCode 方法返回两个看似随机的数字而不是两个相等的数字 合同要求的数字。
因为两个逻辑上相等的实例的哈希码变得不相等,如果您尝试搜索一个而另一个在集合中,则最终会查找错误的哈希桶,从而导致null
。
答案 2 :(得分:2)
您的Pair类需要根据hashCode()
的Javadoc中指定的合同实施equals()
和Object
。