使用泛型类型构造一个ObservableList

时间:2018-09-02 14:40:38

标签: java tableview type-inference generic-type-argument

我正在尝试扩展JavaFX TableView类,以便可以从字面上传递一个Entity类,它将显示该类的所有保存对象。我可以使它像这样工作,但是我希望我的自定义类足够通用,只接受Class变量并负责其余的工作:

private void showSuppliers()
{
    tabPane.getTabs().stream().filter((tab) -> (tab.getText().compareTo("Fournisseurs") == 0)).map((tab) -> (VBox)tab.getContent()).forEachOrdered((vBox) -> 
    {
        TableView<Supplier> table = new TableView<>();

        TableColumn nameColumn = new TableColumn("Nom");
        nameColumn.setCellValueFactory(new PropertyValueFactory<>("name"));

        TableColumn addressColumn = new TableColumn("Addresse");
        addressColumn.setCellValueFactory(new PropertyValueFactory<>("address"));

        TableColumn contactColumn = new TableColumn("Contact");
        contactColumn.setCellValueFactory(new PropertyValueFactory<>("contactPerson"));

        TableColumn phoneNumberColumn = new TableColumn("Téléphone");
        phoneNumberColumn.setCellValueFactory(new PropertyValueFactory<>("phoneNumber"));

        TableColumn emailColumn = new TableColumn("Courriel");
        emailColumn.setCellValueFactory(new PropertyValueFactory<>("emailAddress"));

        TableColumn creationDateColumn = new TableColumn("Date de création");
        creationDateColumn.setCellValueFactory(new PropertyValueFactory<>("enteredDate"));

        table.getColumns().addAll(nameColumn, addressColumn, contactColumn, phoneNumberColumn, emailColumn, creationDateColumn);

        ObservableList<Supplier> suppliers = FXCollections.observableArrayList(supplierService.findAll());
        table.setItems(suppliers);

        vBox.getChildren().add(table);
    });
}

同样,上述方法可以完美地工作,但是我将不得不为列出我要避免的不同数据的每个表重复该方法。我的计划是使用通用类型扩展TableView,例如:

public class EntityTable<T> extends TableView<T>
{
    private final Class<T> entityClass;

    public EntityTable(Class<T> entityClass) 
    {
        this.entityClass = entityClass;

        init();
    }

    private void init()
    {
        for (Field field : entityClass.getDeclaredFields())
        {
            if ((field.getType() != Set.class) && (field.getName().compareTo("id") != 0))
            {
                TableColumn column = new TableColumn(field.getName());
                column.setCellValueFactory(new PropertyValueFactory<>(field.getName()));

                getColumns().add(column);
            }
        }
    }
}

如您所见,添加列不是问题。我不明白的是如何使用我的T来生成类似ObservableList的内容:

ObservableList<Supplier> suppliers = FXCollections.observableArrayList(supplierService.findAll());

我尝试仅将<Supplier>替换为<T><Class<T>>,但这似乎无效。当这些东西几乎不存在时,我就回溯到Java了,说实话,我在网上发现的信息有点令人困惑。

如何使用T变量作为类型来构造列表?

谢谢!

1 个答案:

答案 0 :(得分:0)

嗯,这很简单,我将其完全删除了!这是最终结果,只需传递一个实体类和一个列表,表将负责其余的工作!

public class EntityTable<T> extends TableView<T>
{
    private final Class<T> entityClass;

    private final List<T> entityList;

    public EntityTable(Class<T> entityClass, List<T> entityList) 
    {
        this.entityClass = entityClass;
        this.entityList = entityList;

        init();
    }

    private void init()
    {
        for (Field field : entityClass.getDeclaredFields())
        {
            if ((field.getType() != Set.class) && (field.getName().compareTo("id") != 0))
            {
                TableColumn column = new TableColumn(field.getName());
                column.setCellValueFactory(new PropertyValueFactory<>(field.getName()));

                getColumns().add(column);
            }
        }

        ObservableList list = FXCollections.observableArrayList(entityList);
        setItems(list);
    }
}

不利之处在于,它将在类中显示具有实际属性名称的列标题(并不总是非常用户友好)。欢迎提供任何在列标题中自动添加更好内容的建议!

EDIT

对于我的列名,我最终创建了一个仅带有值参数的自定义批注,名为FriendlyName。现在,我像这样填充列标题:

FriendlyName friendlyName = field.getAnnotation(FriendlyName.class);

TableColumn column = new TableColumn(friendlyName.value());