【问题标题】:ArrayIndexOutOfBoundsException when making recursive call to a JTable对 JTable 进行递归调用时出现 ArrayIndexOutOfBoundsException
【发布时间】:2012-09-19 16:50:31
【问题描述】:

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

背景

我目前正在为大学三年级计算机科学课程做一个小组(我们两个人)编程项目。 该程序的目标是实质上使用电子表格程序来表示 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 行),但我愿意应要求提供。我一直在为此绞尽脑汁,但完全没有找到与此问题相关的任何线程。

【问题讨论】:

  • 天哪,递归 JTables,谈论大量的内存使用。你是说递归方法有2000行长吗?如果是这样,那么某些事情发生了可怕的、可怕的错误。
  • 为什么要扩展JFrame?请编辑您的问题以包含一个 sscce 来展示您所描述的问题。
  • @thatidiotguy 哈哈,不,是整个程序有 2000 行长,递归调用只是对我的构造函数,然后创建并填充 JTable。
  • never-ever 让编辑器做任何事情,但通知其听众它已准备好 - 产生一个线程来获取结果是......非常错误。唯一要做的地方是 getEditorValue 的调用者。此外,像您一样生成线程很可能违反了 Swing 的 EDT 规则(阅读教程中的并发章节)

标签: java xml swing jtable spreadsheet


【解决方案1】:

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

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

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

我会为这些进行调试。但同样,如果没有代码 sn-ps,我会说真的很难回答。

希望这会有所帮助。 :)

【讨论】:

  • 我肯定将正确的表解析到子表中,但我突然想到我可能在错误的地方进行递归调用。不过感谢您的帮助,我添加了一些代码 sn-ps 可能有助于理解问题。
【解决方案2】:

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

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

DefaultTableModel 就足够了。

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

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

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

【讨论】:

    猜你喜欢
    • 2012-10-20
    • 2015-09-08
    • 1970-01-01
    • 1970-01-01
    • 2020-02-07
    • 2017-05-06
    • 2015-03-13
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多