我有一个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)
Run Code Online (Sandbox Code Playgroud)
我的代码出了什么问题?甚至可能我想做什么?
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) {
}
}
Run Code Online (Sandbox Code Playgroud)
小智 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)
Run Code Online (Sandbox Code Playgroud)
说,该DEF声明为list的ColDef<BauwerkOption, ? extends Object>和预计的东西,这延伸的Object,而不是Object自己.所以,如果你? extends从声明中删除DEF,它应该工作
编辑
在你的代码更具体的,你可以创造价值,对象类为Bauwerkstyp,Bezeichnung,Nutzungsart,Durchmesser,等扩展你想要的类型,例如String或Double和实现接口
public ColDefStringValue implements IColDefValue<String> {
@Override
public String getValue() {
return ...;
}
}
Run Code Online (Sandbox Code Playgroud)
界面看起来像这样
public interface ColDefValue<T> {
T getValue();
}
Run Code Online (Sandbox Code Playgroud)
可以使用接口代替? extends Object或Object
private static List<ColDef<BauwerkOption, IColDefValue<?>>> DEF = new ArrayList<>();
Run Code Online (Sandbox Code Playgroud)
没有干净的方法,因为setValueAt无法更改Object参数的类型.
如果您替换以下代码,它将尽可能干净:
这只是为了方便(你不需要自己的lambdas):
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));
}
Run Code Online (Sandbox Code Playgroud)
在ColDef中,只需添加此方法而不更改任何其他内容:
@SuppressWarnings("unchecked")
public BiConsumer<InputObjekt, Object> getObjectValueSetterFunction() {
return (BiConsumer<InputObjekt, Object>) valueSetterFunction;
}
Run Code Online (Sandbox Code Playgroud)
然后将TableModelSelektierung中的setValueAt替换为:
@Override
public void setValueAt(Object aValue, int row, int column) {
if (DEF.get(column).getValueSetterFunction() != null) {
DEF.get(column).getObjectValueSetterFunction().accept(getObjectAt(row), aValue);
}
}
Run Code Online (Sandbox Code Playgroud)
保持您的泛型到位,但解决从继承获得的对象限制.您应该在setValueAt中添加类型检查以确保object参数的类型正确.
| 归档时间: |
|
| 查看次数: |
551 次 |
| 最近记录: |