我有一个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) {
}
}
答案 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
,它应该可以正常工作
修改强>
在代码中更具体,您可以为DEF
,Bauwerkstyp
,Bezeichnung
,Nutzungsart
等创建扩展类型的值对象类你想要的,例如Durchmesser
或String
并实现一个接口
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 ?