4

对于这篇文章的篇幅,我提前道歉,但我已经对我的程序的设计和实现进行了相当深入的描述。

背景

我目前正在为大学三年级计算机科学课程做一个小组(我们两个)编程项目。该程序的目标是实质上使用电子表格程序来表示 XML 文件数据,其中每个 XML 文件都是历史记录。

设计:

电子表格中的每条记录(行)对应于一个 XML 文件,记录的列对应于 XML 文件中的元素。我们通过将单元格组件设置为一个按钮来处理重复的元素(即具有相同标签的元素),单击该按钮时,会打开另一个电子表格,其中包含所有具有重复名称的元素的列表(对于相应的文件)。子元素的处理方式类似,如果一个元素有子元素,则 XML 文件中的相应单元格包含一个按钮,单击该按钮时,会打开一个包含该元素所有子元素的电子表格。

执行:

我们系统的实现是用 Java 编写的。我们的主类(名称为 SpreadSheetGUI)扩展了 JFrame,我们在其中添加了一个 JTable(使用默认表模型)。我们有三种不同的单元格渲染器:一种用于单元格只有文本时,一种用于单元格有按钮时,另一种用于单元格中有文本和按钮时。当单击按钮打开一个新的电子表格(对于子元素或重复的元素名称)时,我们会递归调用电子表格构造函数,然后创建子电子表格。渲染器按以下顺序添加:如果单元格对应于具有多次使用标签的元素,则将按钮添加到单元格中,否则如果单元格对应于具有子节点的元素,我们将添加文本和按钮到单元格,如果单元格只有文本,我们将文本添加到该单元格。

GUI的构造函数是这样的

        /**
     * Parameterised constructor
     * @param dataVector - Vector of vectors of objects that represent the cell values
     * @param columnNames - The vector of objects that represent the column names
     */
    @SuppressWarnings("unchecked")
    public SpreadSheetGUI(Vector<Vector<LevelElementT>> dataVector, Vector<String> columnNames, boolean hasRepeatedColumns, boolean initialFrame)
    {           
        this.hasRepeatedColumns = hasRepeatedColumns;
        this.initialFrame = initialFrame;

        if (initialFrame)
            populateTable(dataVector, columnNames);

        else if (!hasRepeatedColumns)
            populateTable((Vector<Vector<LevelElementT>>)findRepeatedColumns(dataVector).get(0),
                    (Vector<String>)findRepeatedColumns(dataVector).get(1));

        else
            populateTable(dataVector, columnNames);
        //Get repeated column names and add to repeated column hashmap
        //parseElements(dataVector);
    }

其中 populateTable 方法初始化表模型。正如我之前所说,我们的单元格有三个不同的渲染器和编辑器,其中两个渲染器中有按钮,单击时会创建一个新的电子表格(即我们调用电子表格构造函数),例如在我们的一个单元格编辑器具有以下代码

             public Object getCellEditorValue() 
          {
              if (isPushed)
              {
                    //will have the child elements of the current cell element
                Vector<Vector<LevelElementT>> children = new Vector<Vector<LevelElementT>>();
                children.add(new Vector<LevelElementT>());

                List<Element> tempChildren = elements.get(row).get(column).getChildren();

                for (Element child : tempChildren)
                    children.get(0).add(new LevelElementT(child));
                //creates our subspreadsheet
                new Thread(new SpreadSheetGUI(children, new Vector<String>(), false, false)).start();
              }

              isPushed = false;
              return new String(bLabel);
          }

LevelElementT 只是我们创建的一个类,它扩展了覆盖 toString 方法的 Element(在 JDOM2 包中找到)。

问题:

正如我之前提到的,我们创建了一组渲染器来处理向单元格添加按钮,但似乎在创建“子”电子表格时,渲染器正在尝试根据子电子表格编号渲染超出范围的单元格行和列,并抛出数组索引超出范围异常。

更具体地说,错误是在从 populateTable() 方法中获取的以下代码中引发的。我使用 defaultTableModel 的实例初始化 JTable,并设置方法来确定每个组件的渲染器

            table = new JTable(tableModel)
        {
            private static final long serialVersionUID = 1L;

            public TableCellRenderer getCellRenderer(int row, int column)
            {
                //System.out.println(elements.get(row).get(column).getChildren().size());
                if (column == 0)
                {
                    Class<? extends Object> cellClass = getValueAt(row, column).getClass();
                    return getDefaultRenderer(cellClass);
                }
                else if(repeatedColumns.containsKey(table.getColumnModel().getColumn(column).getIdentifier() + " " + elements.get(row).get(0).getText()))
                        return getDefaultRenderer(JButton.class);

                else if(!elements.get(row).get(column).getChildren().isEmpty())
                        return getDefaultRenderer(JPanel.class);

                else 
                        return getDefaultRenderer(JTextArea.class);
            }

            public TableCellEditor getCellEditor(int row, int column)
            {
                if (column == 0)
                {
                    Class<? extends Object> cellClass = getValueAt(row, column).getClass();
                    return getDefaultEditor(cellClass);
                }
                else if(repeatedColumns.containsKey(table.getColumnModel().getColumn(column).getIdentifier() + " " + elements.get(row).get(0).getText()))
                        return getDefaultEditor(JButton.class);

                else if(!elements.get(row).get(column).getChildren().isEmpty())
                        return getDefaultEditor(JPanel.class);

                else 
                        return getDefaultEditor(JTextArea.class);
            }
        };

错误是在 if 语句中引发的(取决于您单击的按钮),这绝对不是因为元素(levelElements 的二维向量)或重复列(以字符串为键和元素向量的哈希表)价值)。


我猜这个问题源于我们对电子表格构造函数进行递归调用这一事实。也有朋友提出这个问题可能是默认表模型导致的,要不要考虑创建自定义表模型?

我没有包含我的代码,因为它相当长(总共大约 2000 行),但我愿意根据要求提供。我一直在为此绞尽脑汁,但完全没有找到与此问题相关的任何线索。

4

2 回答 2

2

由于没有提供代码,我只是猜测:

  • 如果递归调用没有正确返回,它可能会导致这种情况。我们不知道你到底在做什么...

  • 这种现象也可能是由于意外将父表的单元格传递给渲染器,而不是子表的单元格。

我会为这些调试。但同样,如果没有代码片段,我会说真的很难回答。

希望这会有所帮助。:)

于 2012-09-19T17:34:37.400 回答
0

new SpreadSheetGUI(children, new Vector<String>(), false, false)是可疑的(仅猜测),因为之后新的 Vector 无法正常访问。

没有.get(row).get(column)关于索引检查的规定。特别是每一行是否有足够的字符串。

DefaultTableModel 就足够了。

Vector虽然是线程安全的,但已经很老了;使用具有并发性的 List / ArrayList 给人留下更好的印象:CopyOnWriteArrayList

已经存在 XML 表模型和树表模型。我不知道您对编程的要求。

堆栈跟踪、断点、调试、IDE 应该会有所帮助。NetBeans IDE 很容易,eclipse IDE 普及更广。

于 2012-09-24T13:31:41.113 回答