IdentitySet / IdentityHashSet (используйте IdentityHashMap)

Я знаю об IdentityHashMap, но мне нужно использовать что-то вроде "identitySet" (которое использует равно -> o1 == o2). Я собираюсь использовать один, когда я слушаю список наблюдаемых с помощью "экстрактора" (JavaFX):

 List<Person> deleteList = new ArrayList<>();
 List<Person> addList = new ArrayList<>();

 ObservableList<Person> list = FXCollections.observableArrayList(Person.extractor());
    list.add(new Person("a",1));
    list.add(new Person("b",2));
    list.add(new Person("c",3));

    list.addListener((ListChangeListener<Person>) observable -> {
        if(observable.next()) {
            if (observable.wasAdded()) {
                addList.addAll(observable.getAddedSubList());
            }
            if (observable.wasUpdated()) {
                deleteList.add(list.get(observable.getFrom()));
            }
            if (observable.wasRemoved()) {
                deleteList.addAll(observable.getRemoved());
            }
        }
    });

Персональный класс:

public class Person {
private final StringProperty s;
private final DoubleProperty d;

//Getters and Setters
// ...

@Override
public int hashCode() {
    int result = s != null ? s.getValue().hashCode() : 0;
    result = 21 * result + (d != null ? Double.hashCode(d.getValue()) : 0);
    return result;
}

@Override
public boolean equals(Object o) {
    if (this == o) return true;
    if (o == null || getClass() != o.getClass()) return false;

    Person person = (Person) o;
    if (!s.getValue().equals(person.s.getValue())) return false;
    return d.getValue().equals(person.d.getValue());
}


/**
 *Extractor to observe changes in "Property" fields.
 * @return extractor
 */
public static Callback<Person, Observable[]> extractor() {
    return (Person p) -> new Observable[]{p.sProperty(), p.dProperty()};
}

Мне нужно переопределить equals/hashCode, чтобы решить другие мои проблемы.

    Person p = new Person("a",1);
    Set<Object> set = new HashSet<>();
    set.add(p);
    p.setD(9999);
    set.add(p);
    System.out.println(set.size());  // size = 2;

2 ответа

Решение

thx, @BoristheSpider => get Set, который оборачивает IdentityHashMap, используйте Collections.newSetFromMap ():

// get IdentytitySet wich wrap IdentityHashMap
Set<Person> set = Collections.newSetFromMap( new IdentityHashMap<>() ) 
//test 
Person p  = new Person("a",1);
Person p2 = new Person("a",1);
set.add(p);
set.add(p2);
p.setD(999999999);
set.add(p);//add Person with changed "d" to 999999999
System.out.println(set.toString());

Распечатать:

[Person {s = StringProperty [значение: a], d=DoubleProperty [значение: 9.99999999E8]}, Person{s=StringProperty [значение: a], d=DoubleProperty [значение: 1.0]}]

Вы можете использовать гуавы Sets.newIdentityHashSet(),

https://github.com/google/guava/wiki/CollectionUtilitiesExplained

Другие вопросы по тегам