Как использовать методы Collections (removeAll () и retainAll ()) для двух объектов.(объекты являются родительско-дочерними отношениями) - PullRequest
4 голосов
/ 16 декабря 2010

Я ожидал результата ниже, но на самом деле нет. Я хотел бы знать, как показать различия между двумя коллекциями. (объектами являются родительские и дочерние отношения) В этом случае я могу использовать стандартный метод, например removeAll (), или вы можете порекомендовать другой подход, такой как использование apache-commons. Спасибо.

CONSTRAINT
------------------------------
1.Item.class is unmodifiable(eg. I can not add equals method)
2.If id is same between two objects, they are assumed as same things.
------------------------------

EXPECTED
------------------------------
removed object are:
2
same object are:
1
3
add object are:
4
------------------------------

ACTUAL
------------------------------
removed object are:
1
2
3
same object are:
add object are:
1
3
4
------------------------------

package com.javastudy;

import java.util.ArrayList;
import java.util.List;

public class CollectionCompareToObjects {

    public static void main(String[] args) {

        List<Item> before = new ArrayList<Item>();
        List<ItemEx> after = new ArrayList<ItemEx>();

        before.add(new Item(1L));
        before.add(new Item(2L)); // delete
        before.add(new Item(3L));

        after.add(new ItemEx(1L));
        after.add(new ItemEx(3L));
        after.add(new ItemEx(4L)); // added

        List<Item> removed = new ArrayList<Item>(before);
        removed.removeAll(after);

        System.out.println("removed objects are:");
        for(Item item : removed){
            System.out.println(item.getId());
        }

        List<Item> same = new ArrayList<Item>(before);
        same.retainAll(after);

        System.out.println("same objects are:");
        for(Item item : same){
            System.out.println(item.getId());
        }

        List<Item> added = new ArrayList<Item>(after);
        added.removeAll(before);

        System.out.println("add objects are:");
        for(Item item : added){
            System.out.println(item.getId());
        }

    }

}


package com.javastudy;

public class Item {

    private Long id;

    public Item(Long id) {
        this.id = id;
    }

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

}

package com.javastudy;

public class ItemEx extends Item {

    private String name;

    public ItemEx(Long id) {
        super(id);
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

}

1 Ответ

9 голосов
/ 16 декабря 2010

Java-коллекции используют методы equals и hashCode (последний используется HashMap s, HashSet s и другими). ​​

Если вы хотите использоватьВозможности структуры данных коллекций Java (например, removeAll, retainAll и т. д.), вам необходимо предоставить объектам правильные реализации equals и hashCode.

Если вы не можете изменитьItem class, вы можете написать класс-оболочку с собственной реализацией equals:

public class ItemWrapper {
    private final Item item;
    public ItemWrapper(Item item) {
        this.item = item;
    }

    public Item getItem() {
        return item;
    }

    @Override
    public boolean equals(Object obj) {
        return obj instanceof ItemWrapper && item.getId().equals(((ItemWrapper) obj).item.getId());
    }

    @Override
    public int hashCode() {
        return item.getId().hashCode();
    }
}

Создать новый ItemWrapper для каждого оригинального Item, сохранить ItemWrapper s в Javaколлекции и использовать необходимые методы (removeAll / retainAll).Затем выполните итерацию по полученной коллекции и извлеките Item s, вызвав каждый ItemWrapper 'getItem() метод.

Другой вариант - создать подкласс ArrayList, но это выглядит как более запутанное решение.

Еще один вариант - не использовать коллекции Java для логики удаления / сохранения, а реализовывать их самостоятельно.

...