让BiConsumer接受对象

时间:2018-04-11 06:24:15

标签: java

我有一个JTable TableModel的列定义列表,列“B”有一个setter BiConsumer,它接受一个BauwerkOption类和一个字符串。

当我尝试使用“... accept ...”设置字符串时,我收到以下错误:

The method accept(Selektierung.BauwerkOption, capture#4-of ? extends Object) in the type BiConsumer<Selektierung.BauwerkOption,capture#4-of ? extends Object> is not applicable for the arguments (Selektierung.BauwerkOption, Object)

我的代码出了什么问题?甚至可能我想做什么?

public class TableModelSelektierung extends DefaultTableModel {
    private static final long serialVersionUID = -5921626198599251183L;
    private List<BauwerkOption> data;
    private static List<ColDef<BauwerkOption, ? extends Object>> DEF = new ArrayList<>();
    static {
        DEF.add(new ColDef<BauwerkOption, String>("A", (o) -> o.getBauwerkstyp()));
        DEF.add(new ColDef<BauwerkOption, String>("B", (o) -> o.getBezeichnung())
                                                                                            .withSetValueAtFunction((i, o) -> i.setBauwerkstyp(o)));
        DEF.add(new ColDef<BauwerkOption, String>("C", (o) -> o.getNutzungsart()));
        DEF.add(new ColDef<BauwerkOption, Double>("D", (o) -> o.getDurchmesser()));
    }

    @Override
    public int getColumnCount() {
        return DEF.size();
    }

    @Override
    public boolean isCellEditable(int row, int column) {
        return DEF.get(row).getValueSetterFunction() == null;
    }

    @Override
    public int getRowCount() {
        if (data != null) {
            return data.size();
        }
        return 0;
    }

    @Override
    public String getColumnName(int column) {
        return DEF.get(column).getTitle();
    }

    public void setData(List<BauwerkOption> data) {
        this.data = data;
        fireTableDataChanged();
    }

    public BauwerkOption getObjectAt(int row) {
        return data.get(row);
    }

    @Override
    public void setValueAt(Object aValue, int row, int column) {
        if (DEF.get(column).getValueSetterFunction() != null) {
            DEF.get(column).getValueSetterFunction().accept(getObjectAt(row), aValue);
        }
    }

    @Override
    public Object getValueAt(int row, int column) {
        BauwerkOption o = getObjectAt(row);
        return DEF.get(column).getValueGetterFunction().apply(o);
    }

}


class ColDef<InputObjekt, OutputValue> {
    private String title;
    private Function<InputObjekt, OutputValue> valueGetterFunction;
    private BiConsumer<InputObjekt, OutputValue> valueSetterFunction;

    public ColDef(String title, Function<InputObjekt, OutputValue> valueGetterFunction) {
        this.title = title;
        this.valueGetterFunction = valueGetterFunction;
    }

    public ColDef<InputObjekt, OutputValue> withSetValueAtFunction(BiConsumer<InputObjekt, OutputValue> valueSetterFunction) {
        this.valueSetterFunction = valueSetterFunction;
        return this;
    }

    public Function<InputObjekt, OutputValue> getValueGetterFunction() {
        return valueGetterFunction;
    }

    public String getTitle() {
        return title;
    }

    public BiConsumer<InputObjekt, OutputValue> getValueSetterFunction() {
        return valueSetterFunction;
    }

}

class BauwerkOption {


    public BauwerkOption() {

    }

    public String getBezeichnung() {
        return "";
    }

    public String getBauwerkstyp() {
        return "";
    }

    public Double getDurchmesser() {

        return 0d;
    }

    public String getNutzungsart() {
        return "todo";
    }



    public void setBauwerkstyp(String s) {

    }
}

3 个答案:

答案 0 :(得分:5)

消息

The method accept(Selektierung.BauwerkOption, capture#4-of ? extends Object) in the type BiConsumer<Selektierung.BauwerkOption,capture#4-of ? extends Object> is not applicable for the arguments (Selektierung.BauwerkOption, Object)

说,DEF被声明为list的{​​{1}}并期望某些内容扩展ColDef<BauwerkOption, ? extends Object>而不是Object本身。因此,如果您从Object的声明中删除? extends,它应该可以正常工作

修改

在代码中更具体,您可以为DEFBauwerkstypBezeichnungNutzungsart等创建扩展类型的值对象类你想要的,例如DurchmesserString并实现一个接口

Double

界面看起来像这样

public ColDefStringValue implements IColDefValue<String> {
    @Override
    public String getValue() {
        return ...;
    }
}

可以使用界面代替public interface ColDefValue<T> { T getValue(); } ? extends Object

Object

答案 1 :(得分:5)

没有干净的方法,因为_中的Object参数的类型无法更改。

如果您更换以下代码,它将尽可能干净:

这只是为了方便(你不需要自己的lambdas):

setValueAt

在ColDef中,只需添加此方法而不更改任何其他内容:

private static List<ColDef<BauwerkOption, ?>> DEF = new ArrayList<>();
static {
    DEF.add(new ColDef<BauwerkOption, String>("A", BauwerkOption::getBauwerkstyp));
    DEF.add(new ColDef<BauwerkOption, String>("B", BauwerkOption::getBezeichnung).withSetValueAtFunction(BauwerkOption::setBauwerkstyp));
    DEF.add(new ColDef<BauwerkOption, String>("C", BauwerkOption::getNutzungsart));
    DEF.add(new ColDef<BauwerkOption, Double>("D", BauwerkOption::getDurchmesser));
}

然后将TableModelSelektierung中的setValueAt替换为:

@SuppressWarnings("unchecked")
public BiConsumer<InputObjekt, Object> getObjectValueSetterFunction() {
    return (BiConsumer<InputObjekt, Object>) valueSetterFunction;
}

保持您的泛型到位,但解决从继承获得的对象限制。您应该在setValueAt中添加类型检查,以确保对象参数的类型正确。

答案 2 :(得分:3)

问题在于ColDef#getValueSetterFunction()。这会返回BiConsumer<BauwerkOption, ?>,其中问号指的是从Object延伸的某种类型。没有办法知道这是哪种特定类型,但绝对不是Object本身。将Object类型的值传递给accept中的TableModelSelektierung#setValueAt()方法将因此失败:

incompatible types: java.lang.Object cannot be converted to capture#1 of ?