1

试图为这个问题设计一个解决方案让我不知所措——这是缺乏经验的副产品。

我的目标是读取 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 部分中的值总是唯一的。Name 部分中的值总是唯一的。架构将(据说)永远不会改变。将来,新的唯一值可能会出现在“种类”部分。

我提出的解决方案:创建一个类 XMLNode 来表示来自 XML 源的节点。XMLNode 类的成员对应于每个 Node 中的标签。遍历所有节点并为每个节点创建一个 XMLNode。遍历节点时:在哈希映射中添加 XMLNode 对象,Keys = XMLNode.ID 和 vals = XMLNode。创建一组独特的种类。

从 Kind 条目数组中填充组合框一。从每个 的名称数据中填充组合框两个。

这是一种合适的方法,还是我忽略了更好/更容易/更优雅的解决方案?如果我走在正确的轨道上,我提出的解决方案是否有任何明显的缺陷?

4

2 回答 2

1

代码

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);
    }
}
于 2012-10-16T16:27:12.223 回答
0

我最终得到了一个类 NodeImporter 和一个类 Node。Node代表一个节点,NodeImporter解析XML。在 NodeImporter 中,XML 源被解析并表示为 HashMap 的 HashMap。外层 HashMap (Key,Value) 为 (Kind,(HashMap(Key,Value))。内层 HashMap (Key,Value) 为 (UinqueID,Node),最终结果为 (Kind,(UniqueID,Node))。我将最终结果称为“filteredMap”。NodeImporter 的所有字段和方法都是私有的,除了构造函数和过滤映射的 getter。需要数据来构建组合框的类从 NodeImporter 的实例获取过滤映射。它可以然后获取外部 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;
        }
    }
于 2012-10-25T15:41:28.140 回答