gpt4 book ai didi

java - 具有可比较键的 HashMap 无法按预期工作

转载 作者:行者123 更新时间:2023-11-29 03:01:54 26 4
gpt4 key购买 nike

我们正面临 HashMap 行为方式的奇怪问题。

当HashMap keys实现Comparable接口(interface)但是compareTo实现与equals不一致时HashMaps:

  • 长得比他们应该长的大得多
  • 它们包含多个相同元素的实例
  • 附加到这些元素的值可能不同
  • get(key) 结果取决于使用哪个键(即使键根据 equals 方法相等)。

我创建了一个小测试来重现该问题(见下文)。

import java.util.HashMap;
import java.util.Map;

public class HashMapTest {
private static final char MIN_NAME = 'A';
private static final char MAX_NAME = 'K';
private static final int EXPECTED_NUMBER_OF_ELEMENTS = MAX_NAME - MIN_NAME + 1;

private HashMap<Person, Integer> personToAgeMap;

HashMapTest(){
personToAgeMap = new HashMap();
}

public static void main(String[] args){
HashMapTest objHashMap = new HashMapTest();
System.out.println("Initial Size of Map: " + objHashMap.getPersonToAgeMap().size());
objHashMap.whenOverridingEqualElements_thenSizeOfTheMapIsStable();
objHashMap.whenGettingElementUsingPersonOfAge1_thenOverridenValuesAreReturned();
objHashMap.whenGettingElementUsingPersonOfAge100_thenOverridenValuesAreReturned();
objHashMap.whenGettingElementUsingPersonOfAge50_thenOverridenValuesAreReturned();
objHashMap.whenGettingElementUsingPersonOfAgeMinus1_thenOverridenValuesAreReturned();
}

public HashMap<Person, Integer> getPersonToAgeMap(){
return personToAgeMap;
}

public void whenOverridingEqualElements_thenSizeOfTheMapIsStable() {
System.out.println("Adding elements with age 1..");
putAllPeopleWithAge(personToAgeMap, 1);
System.out.println(personToAgeMap);
System.out.println("Expected Number Of elements: " + EXPECTED_NUMBER_OF_ELEMENTS+ "\nActual Number of elements: "+personToAgeMap.size());

System.out.println();
System.out.println("Overwriting map, with value 100..");
putAllPeopleWithAge(personToAgeMap, 100);
System.out.println(personToAgeMap);
System.out.println("Expected Number Of elements: " + EXPECTED_NUMBER_OF_ELEMENTS+ "\nActual Number of elements: "+personToAgeMap.size());
System.out.println();
}


public void whenGettingElementUsingPersonOfAge1_thenOverridenValuesAreReturned() {
useAgeToCheckAllHashMapValuesAre(1, 100);
}


public void whenGettingElementUsingPersonOfAge100_thenOverridenValuesAreReturned() {
useAgeToCheckAllHashMapValuesAre(100, 100);
}

public void whenGettingElementUsingPersonOfAge50_thenOverridenValuesAreReturned() {
useAgeToCheckAllHashMapValuesAre(50, 100);
}


public void whenGettingElementUsingPersonOfAgeMinus1_thenOverridenValuesAreReturned() {
useAgeToCheckAllHashMapValuesAre(-10, 100);
}

private void useAgeToCheckAllHashMapValuesAre(int age, Integer expectedValue) {
System.out.println("Checking the values corresponding to age = " + age);
StringBuilder sb = new StringBuilder();

int count = countAllPeopleUsingAge(personToAgeMap, age);
System.out.println("Count of People with age " + age+" =" + count);

if (EXPECTED_NUMBER_OF_ELEMENTS != count) {
sb.append("Size of the map ").append(" is wrong: ")
.append("expected <").append(EXPECTED_NUMBER_OF_ELEMENTS).append("> actual <").append(count).append(">.\n");
}

for (char name = MIN_NAME; name <= MAX_NAME; name++) {
Person key = new Person(name, age);
Integer value = personToAgeMap.get(key);
if (!expectedValue.equals(value)) {
sb.append("Unexpected value for ").append(key).append(": ")
.append("expected <").append(expectedValue).append("> actual <").append(value).append(">.\n");
}
}

if (sb.length() > 0) {
System.out.println(sb.toString());
}
}

void putAllPeopleWithAge(Map<Person, Integer> map, int age) {
for (char name = MIN_NAME; name <= MAX_NAME; name++) {
map.put(new Person(name, age), age);
}
}

int countAllPeopleUsingAge(Map<Person, Integer> map, int age) {
int counter = 0;
for (char name = MIN_NAME; name <= MAX_NAME; name++) {
if (map.containsKey(new Person(name, age))) {
counter++;
}
}
return counter;
}

String getAllPeopleUsingAge(Map<Person, Integer> map, int age) {
StringBuilder sb = new StringBuilder();
for (char name = MIN_NAME; name <= MAX_NAME; name++) {
Person key = new Person(name, age);
sb.append(key).append('=').append(map.get(key)).append('\n');
}
return sb.toString();
}

class Person implements Comparable<Person> {
char name;
int age;

public Person(char name, int age) {
this.name = name;
this.age = age;
}

//Making sure all elements end up in the very same bucket
//Nothing wrong with it except performance...
@Override
public int hashCode() {
return 0;
}

//equals is only by name
@Override
public boolean equals(Object other) {
Person otherPerson = (Person)other;
return this.name == otherPerson.name;
}

public String toString() {
return name + "[age=" + age + "]";
}

//comparing by age
//NOTE: compareTo is inconsistent with equals which should be OK in non-sorted collections
//https://docs.oracle.com/javase/8/docs/api/java/lang/Comparable.html
@Override
public int compareTo(Person other) {
return this.age - other.age;
}
}
}

预期输出

  Initial Size of Map: 0
Adding elements with age 1..
{K[age=1]=1, J[age=1]=1, I[age=1]=1, H[age=1]=1, G[age=1]=1, F[age=1]=1, E[age=1]=1, D[age=1]=1, C[age=1]=1, B[age=1]=1, A[age=1]=1}
Expected Number Of elements: 11
Actual Number of elements: 11

Overwriting map, with value 100..
{K[age=1]=100, J[age=1]=100, I[age=1]=100, H[age=1]=100, G[age=1]=100, F[age=1]=100, E[age=1]=100, D[age=1]=100, C[age=1]=100, B[age=1]=100, A[age=1]=100}
Expected Number Of elements: 11
Actual Number of elements: 11

Checking the values corresponding to age = 1
Count of People with age 1 =11
Checking the values corresponding to age = 100
Count of People with age 100 =11
Checking the values corresponding to age = 50
Count of People with age 50 =11
Checking the values corresponding to age = -10
Count of People with age -10 =11

实际输出

Initial Size of Map: 0
Adding elements with age 1..
{I[age=1]=1, A[age=1]=1, B[age=1]=1, C[age=1]=1, D[age=1]=1, E[age=1]=1, F[age=1]=1, G[age=1]=1, H[age=1]=1, J[age=1]=1, K[age=1]=1}
Expected Number Of elements: 11
Actual Number of elements: 11

Overwriting map, with value 100..
{I[age=1]=100, A[age=1]=1, B[age=1]=100, C[age=1]=1, D[age=1]=100, A[age=100]=100, C[age=100]=100, E[age=100]=100, H[age=100]=100, J[age=100]=100, K[age=100]=100, F[age=100]=100, G[age=100]=100, E[age=1]=1, F[age=1]=1, G[age=1]=1, H[age=1]=1, J[age=1]=1, K[age=1]=1}
Expected Number Of elements: 11
Actual Number of elements: 19

Checking the values corresponding to age = 1
Count of People with age 1 =11
Unexpected value for E[age=1]: expected <100> actual <1>.
Unexpected value for F[age=1]: expected <100> actual <1>.
Unexpected value for G[age=1]: expected <100> actual <1>.
Unexpected value for J[age=1]: expected <100> actual <1>.
Unexpected value for K[age=1]: expected <100> actual <1>.

Checking the values corresponding to age = 100
Count of People with age 100 =10
Size of the map is wrong: expected <11> actual <10>.
Unexpected value for B[age=100]: expected <100> actual <null>.

Checking the values corresponding to age = 50
Count of People with age 50 =5
Size of the map is wrong: expected <11> actual <5>.
Unexpected value for B[age=50]: expected <100> actual <null>.
Unexpected value for E[age=50]: expected <100> actual <null>.
Unexpected value for F[age=50]: expected <100> actual <null>.
Unexpected value for G[age=50]: expected <100> actual <null>.
Unexpected value for J[age=50]: expected <100> actual <null>.
Unexpected value for K[age=50]: expected <100> actual <null>.

Checking the values corresponding to age = -10
Count of People with age -10 =4
Size of the map is wrong: expected <11> actual <4>.
Unexpected value for A[age=-10]: expected <100> actual <1>.
Unexpected value for B[age=-10]: expected <100> actual <null>.
Unexpected value for C[age=-10]: expected <100> actual <1>.
Unexpected value for D[age=-10]: expected <100> actual <null>.
Unexpected value for E[age=-10]: expected <100> actual <1>.
Unexpected value for F[age=-10]: expected <100> actual <null>.
Unexpected value for G[age=-10]: expected <100> actual <null>.
Unexpected value for H[age=-10]: expected <100> actual <null>.
Unexpected value for J[age=-10]: expected <100> actual <null>.
Unexpected value for K[age=-10]: expected <100> actual <null>.

最佳答案

从 Java 8 开始,HashMap 中添加了一项优化,以处理当键为 Comparable 时的冲突:

To ameliorate impact, when keys are java.lang.Comparable, this class may use comparison order among keys to help break ties.

如果您的键具有与 equals() 不一致的自然顺序,则可能会发生许多奇怪的事情,在这种情况下, map 似乎正在尝试使用以下方法跟踪键自然顺序。

关于java - 具有可比较键的 HashMap 无法按预期工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34305067/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com