Копировать ObservableList java

У меня есть MasterData, который является ObservableList и filterData, который также является ObservableList.

Затем я хочу использовать его, чтобы показать отфильтрованные данные, когда фильтр установлен, но также иметь возможность восстанавливать при необходимости, вот MCVE:

package br;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;

public class Main{
    private static ObservableList<Foo> masterData = FXCollections.observableArrayList();
    private static ObservableList<Foo> filteredData = FXCollections.observableArrayList();
    static Filter filter = new Filter();
    public static void addDummy(){
        masterData.add(new Foo("a",1));
        masterData.add(new Foo("aa",3));
        masterData.add(new Foo("b",2));
        masterData.add(new Foo("bb",4));
        masterData.add(new Foo("c",3));
    }
    public static void printData(ObservableList<Foo> list){
        for(Foo f: list) System.out.println(f.getName());
    }
    public static void main(String[] args) {
        addDummy();
        applyFilter(3);
        printData(filteredData);
        applyFilter(0);
        printData(filteredData);
    }
    public static void applyFilter(int num){
        filter.setData(masterData);
        filter.setFilter(num);
        filteredData = filter.getData();
    }
}

Фильтр класса, который я использую для фильтрации данных:

package br;

import java.util.ArrayList;
import java.util.List;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;

public class Filter {
    private ObservableList<Foo> data = FXCollections.observableArrayList();
    public void setFilter(int filter){
        List<Foo> copy = new ArrayList<Foo>(data);
        for(Foo f: copy)
            if(f.getBar()<filter) data.remove(f);
    }
    public ObservableList<Foo> getData() {
        return data;
    }

    public void setData(ObservableList<Foo> data) {
        this.data = data;
    }

}

И класс Foo, который является просто строковым именем и строкой int (плюс getters/seters)

package br;

public class Foo {
    private String name;
    private int bar;
    public Foo(String name, int bar){
        this.setBar(bar);
        this.setName(name);
    }
    public String getName() {return name;}
    public void setName(String name) {this.name = name;}
    public int getBar() {return bar;}
    public void setBar(int bar) {this.bar = bar;}
}

Когда вы запустите этот код, вы получите:

аа
бб
с

Оба, когда фильтр был установлен на 3 (, который отфильтровал данные как ожидалось), и после того, как фильтр был "сброшен" до нуля.

Как я могу убедиться, что фильтр всегда обрабатывается с данными в masterData​​strong > , а затем хранится в отфильтрованном виде?

Примечание: это для проекта JavaFX, поэтому мне действительно нужно использовать ObservableList.

Ответ 1

Проблема

public void setData(ObservableList<Foo> data) {
    this.data = data;
}

Объяснение

Если вам интересно, что Java подчиняется Pass By Value, а значение masterData​​strong > должно быть скопировано в список data​​strong > . Ты прав! Но есть лишь немного поворот к истории!

Включение примитивов, значения передаются, но выводятся объекты, ссылки на объекты передаются по значению.

Итак, вы ссылаетесь на Object masterData​​strong > , вместо того, чтобы копировать значения в новый List!

Решение

this.data = FXCollections.observableArrayList(data);

Это создаст новый объект для данных для ссылки, и все манипуляции будут выполняться с помощью этих новых данных объекта.