对于这篇文章的篇幅,我提前道歉,但我已经对我的程序的设计和实现进行了相当深入的描述。
背景
我目前正在为大学三年级计算机科学课程做一个小组(我们两个)编程项目。该程序的目标是实质上使用电子表格程序来表示 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 行),但我愿意根据要求提供。我一直在为此绞尽脑汁,但完全没有找到与此问题相关的任何线索。