我为尝试为这个问题设计解决方案而感到不知所措 - 这是缺乏经验的副产品。
我的目标是读取XML输入文件,存储XML中的信息,并使用XML中的数据填充两个组合框。第二个组合框的内容将根据第一个组合框中的选择进行更改。
鉴于此XML结构:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<Category xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<Node>
<ID>Unique string</ID>
<Name>Unique string</Name>
<Code>Generic string<Code>
<Kind>Generic string</Kind>
<Frame>Generic string</Frame>
...
</Node>
...
</Category>
第一个组合框: 必须仅包含在“种类”部分中找到的唯一值。
第二个组合框: 包含来自每个节点的所有名称条目,其种类等于在第一个组合框中选择的种类。
关于XML源: 它是外部维护和生成的。 ID部分中的值始终是唯一的。 “名称”部分中的值始终是唯一的。 架构(假设)永远不会改变。 未来,“种类”部分中可能会出现新的唯一值。
我建议的解决方案: 创建一个类XMLNode以表示XML源中的节点。 XMLNode类的成员对应于每个Node中的标记。 遍历所有节点并为每个节点创建一个XMLNode。 循环通过节点时: 在哈希映射中添加XMLNode对象,其中Keys = XMLNode.ID,vals = XMLNode。 创建一个独特的种类数组。
从Kind条目数组中填充组合框1。 从每个的名称数据中填充组合框2。
这是一种合适的方法,还是我忽略了更好/更容易/更优雅的解决方案? 如果我走在正确的轨道上,我建议的解决方案是否有任何明显的缺陷?
答案 0 :(得分:1)
一ll updates to the JComboBox and its ComboBoxModel must be done on EDT
将所有FileIO
或Database
个事件重定向至Runnable#Thread
或SwingWorker
也许这个逻辑可以帮到你,
代码
import java.awt.*;
import java.awt.event.*;
import java.util.*;
import javax.swing.*;
public class ComboBoxTwo extends JFrame implements ActionListener, ItemListener {
private static final long serialVersionUID = 1L;
private JComboBox mainComboBox;
private JComboBox subComboBox;
private Hashtable<Object, Object> subItems = new Hashtable<Object, Object>();
public ComboBoxTwo() {
String[] items = {"Select Item", "Color", "Shape", "Fruit"};
mainComboBox = new JComboBox(items);
mainComboBox.addActionListener(this);
mainComboBox.addItemListener(this);
//prevent action events from being fired when the up/down arrow keys are used
//mainComboBox.putClientProperty("JComboBox.isTableCellEditor", Boolean.TRUE);
getContentPane().add(mainComboBox, BorderLayout.WEST);
subComboBox = new JComboBox();// Create sub combo box with multiple models
subComboBox.setPrototypeDisplayValue("XXXXXXXXXX"); // JDK1.4
subComboBox.addItemListener(this);
getContentPane().add(subComboBox, BorderLayout.EAST);
String[] subItems1 = {"Select Color", "Red", "Blue", "Green"};
subItems.put(items[1], subItems1);
String[] subItems2 = {"Select Shape", "Circle", "Square", "Triangle"};
subItems.put(items[2], subItems2);
String[] subItems3 = {"Select Fruit", "Apple", "Orange", "Banana"};
subItems.put(items[3], subItems3);
// mainComboBox.setSelectedIndex(1);
}
@Override
public void actionPerformed(ActionEvent e) {
String item = (String) mainComboBox.getSelectedItem();
Object o = subItems.get(item);
if (o == null) {
subComboBox.setModel(new DefaultComboBoxModel());
} else {
subComboBox.setModel(new DefaultComboBoxModel((String[]) o));
}
}
@Override
public void itemStateChanged(ItemEvent e) {
if (e.getStateChange() == ItemEvent.SELECTED) {
if (e.getSource() == mainComboBox) {
if (mainComboBox.getSelectedIndex() != 0) {
FirstDialog firstDialog = new FirstDialog(ComboBoxTwo.this,
mainComboBox.getSelectedItem().toString(), "Please wait, Searching for ..... ");
}
}
}
}
private class FirstDialog extends JDialog {
private static final long serialVersionUID = 1L;
FirstDialog(final Frame parent, String winTitle, String msgString) {
super(parent, winTitle);
setModalityType(Dialog.ModalityType.APPLICATION_MODAL);
JLabel myLabel = new JLabel(msgString);
JButton bNext = new JButton("Stop Processes");
add(myLabel, BorderLayout.CENTER);
add(bNext, BorderLayout.SOUTH);
bNext.addActionListener(new ActionListener() {
@Override
public void actionPerformed(ActionEvent evt) {
setVisible(false);
}
});
javax.swing.Timer t = new javax.swing.Timer(1000, new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
setVisible(false);
}
});
t.setRepeats(false);
t.start();
setLocationRelativeTo(parent);
setSize(new Dimension(400, 100));
setVisible(true);
}
}
public static void main(String[] args) {
JFrame frame = new ComboBoxTwo();
frame.setDefaultCloseOperation(EXIT_ON_CLOSE);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
}
答案 1 :(得分:0)
我最终得到了一个NodeImporter类和一个Node类。 Node表示节点,NodeImporter解析XML。在NodeImporter中,XML源被解析并表示为HashMaps的HashMap。外部HashMap(Key,Value)是(Kind,(HashMap(Key,Value))。内部HashMap(Key,Value)是(UinqueID,Node),用于(Kind,(UniqueID,Node))的最终结果。我将最终结果称为“filteredMap”。所有NodeImporter的字段和方法都是私有的,除了构造函数和filterMap的getter。 需要数据来构建组合框的类从NodeImporter的实例获取filteredMap。然后它可以获取外部HashMap的键来构建第一个ComboBoxModel。它可以很容易地将内部HashMap用作第二个组合框的HashMap。
设置的伪代码:
使用类成员构建的NodeImporter类:
arrayList(String) uniqueKinds = null
arrayList(Node) allNodes = null
HashMap(String, HashMap(String,Node))) filteredNodes = null
类NodeImporter构造函数:
open XML
while xml source has next
{
add next node to allNodes, key = node.uniqueId, Val = node
if next node.kind not in uniqueKinds, add node.kind to uniqueKinds
}
ClassNodeImporter method makeFilteredeMap:
private boolean makeFilteredeMap() {
if (uniqueKinds.isEmpty()) {
return false;
} else {
for (String k : uniqueKinds) {
HashMap<String, Node> aMap = new HashMap<String, Node>();
for (Node n : allNodes) {
if (n.getKind().equals(k)) {
aMap.put(n.getCode(), n);
}
}
filteredNodes.put(k, aMap);
}
return true;
}
}