在previous问题解决之后,我可以使用一个SwingWorker
使用HashMap
填充多个组合框。现在我想通过使用一个映射(如果可能)来填充/更新gui应用程序的不同面板中包含的多个组合框和列表AND表。在javadoc中,我看到DefaultListModel
和DefaultComboBoxModel
都延伸AbstractListModel
,所以我改变了地图
Map<String, DefaultComboBoxModel>
到
Map<String, AbstractListModel>
然后传递给SwingWorker
。以下代码有效,它可以使用这一个映射填充许多组合和列表。
SSCCE:
public class TestPanel extends JPanel {
private final Map<String, AbstractListModel> map = new HashMap<String, AbstractListModel>();
private TestPanel() {
setLayout(new BoxLayout(this, BoxLayout.Y_AXIS));
JComboBox combo1 = new JComboBox();
JComboBox combo2 = new JComboBox();
JList list1 = new JList(new DefaultListModel());
JList list2 = new JList(new DefaultListModel());
add(combo1);
add(combo2);
add(new JLabel("LIST 1:"));
add(list1);
add(new JLabel("LIST 2:"));
add(list2);
map.put("ComboBox1", (DefaultComboBoxModel)combo1.getModel());
map.put("ComboBox2", (DefaultComboBoxModel)combo2.getModel());
map.put("List1", (DefaultListModel)list1.getModel());
map.put("List2", (DefaultListModel)list2.getModel());
new MyWorker(map).run();
}
public static void main(String[] args) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
JFrame frame = new JFrame();
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setContentPane(new TestPanel());
frame.setSize(200, 300);
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
private class MyWorker extends SwingWorker<Void, Object[]> {
private final int COMBO_BOX_MODEL = 1;
private final int LIST_MODEL = 2;
private final Map<String, AbstractListModel> map;
public MyWorker(Map<String, AbstractListModel> map) {
this.map = map;
}
@Override
protected Void doInBackground() throws Exception {
// just a random way to fill the combos and the lists
for (int i = 0; i < 20; i++) {
Object[] cell = new Object[3];
if (i>=0 && i<=4) {
cell[0] = "ComboBox1";
cell[1] = COMBO_BOX_MODEL;
} else if (i>=5 && i<=9) {
cell[0] = "ComboBox2";
cell[1] = COMBO_BOX_MODEL;
} else if (i>=10 && i<=14) {
cell[0] = "List1";
cell[1] = LIST_MODEL;
} else if (i>=15 && i<=20) {
cell[0] = "List2";
cell[1] = LIST_MODEL;
}
cell[2] = "value " + i;
publish(cell);
}
return null;
}
@Override
protected void process(List<Object[]> chunks) {
for (Object[] chunk : chunks) {
if (chunk[1] == Integer.valueOf(COMBO_BOX_MODEL)) {
DefaultComboBoxModel model = (DefaultComboBoxModel)map.get(chunk[0]);
model.addElement(chunk[2]);
}
else if (chunk[1] == Integer.valueOf(LIST_MODEL)) {
DefaultListModel model = (DefaultListModel)map.get(chunk[0]);
model.addElement(chunk[2]);
}
}
}
};
}
问题1:虽然上面的代码有效,但我不喜欢它的外观(尤其是process()
方法中的检查)。有没有更好的和/或更聪明的方法来完成整个事情?
问题2:有没有办法让SwingWorker
使用一个地图来更新表格以及组合和列表?我找不到一种方法,因为DefaultTableModel
扩展AbstractTableModel
,这与地图使用的AbstractListModel
不同。
我希望我很清楚,任何帮助都会受到赞赏。
答案 0 :(得分:4)
您的数据源无疑具有与Java不同的类型系统;将这些类型与Java类型相匹配的代码不会更漂亮,但至少你已经封装了它。 enum
可能会有所帮助。
是的,您可以通过委派默认实现来扩展AbstractTableModel
并实现其他接口。有一个大纲here。您必须确定这样的瓶颈是否适用于您的应用程序。
public class SharedModel extends AbstractTableModel
implements ComboBoxModel, ListModel
附录:另一种方法是编写三个模型(MyTableModel
,MyListModel
,MyComboBoxModel
),每个模型都引用一个公共数据模型。
您的意思是由三个模型实现的通用接口吗?
是的,由类({1}}实现的接口,它将数据库读入内部格式。然后,客户端模型可以向ApplicationDataModel
询问每个需要履行其自身接口合同的数据。