0

我有一个模块类型的ArrayList,每个模块都有一个Assignments 的arraylist。我已经编写了以下 TableModel,但是当在表中选择某些内容时,我遇到了问题,这会导致我的 ArrayLists 超出范围。这是我的表模型:

public class AssignmentsTableModel extends AbstractTableModel {
private ArrayList<Module> modules;
private static final String[] COLUMN_NAMES = {"Module Identifier", "Module Name", "Assignment Title", "Author", "Date Set", "Date Due", "Weighting"};

private int moduleID;
private int assignmentID;
private Module module;
private int totalNumberAssignments;

public AssignmentsTableModel(ArrayList<Module> modules) {
    moduleID = 0;
    assignmentID = 0;
    this.modules = modules;
    module = modules.get(moduleID);
    totalNumberAssignments = getRowCount();
}

public AssignmentsTableModel(Module module) {
    modules = new ArrayList<Module>();
    modules.add(module);
}

@Override
public int getRowCount() {
    int rowCount = 0;
    for(Module mod : modules){
        rowCount += mod.getAssignments().size();
    }
    return rowCount;
}

@Override
public int getColumnCount() {
    return COLUMN_NAMES.length;  //To change body of implemented methods use File | Settings | File Templates.
}

@Override
public Object getValueAt(int row, int column) {
    if(column%getColumnCount() == 0 && getRowCount() <= totalNumberAssignments){
        if(isAtLastAssignment(row, moduleID)){
            assignmentID = 0;
            moduleID += 1;
        }
    }

    module = modules.get(moduleID);
    ArrayList<Assignment> assignments = module.getAssignments();

    Assignment assignment = assignments.get(assignmentID);

    switch (column){
        case 0: return module.getIdentifier();
        case 1: return module.getTitle();
        case 2: return assignment.getTitle();
        case 3: return assignment.getAuthor();
        case 4: return assignment.getSet();
        case 5: return assignment.getDue();
        case 6: return assignment.getWeighting();
        default: return null;
    }


}

public String getColumnName(int columnIndex) {
    return COLUMN_NAMES[columnIndex];
}

public boolean isAtLastAssignment(int row, int moduleID){
    boolean atLast = false;
    ArrayList<Assignment> assignments = modules.get(moduleID).getAssignments();
    if(moduleID == 0){
        if(row == assignments.size()){
            atLast = true;
        }
    } else {
        int assignmentsSize = assignments.size() + getCurrentRowNumber(moduleID);
        if(row == assignmentsSize){
            atLast = true;
        }
    }

    return atLast;
}

public int getCurrentRowNumber(int moduleID){
    int rowNumber = 0;

    for(int i = 0; i < moduleID; i++){
        rowNumber = modules.get(i).getAssignments().size()-1;
    }

    return rowNumber;
}
}

如您所见,我将 moduleID 存储为全局变量,这在它第一次启动时很好,但是一旦选择了某些东西,它就会使用最后一个值。我还能做些什么来阻止这种情况发生?

4

2 回答 2

5

TableModel 方法getValueAt()由表渲染器每次想要获取单元格的值时调用,以便它可以渲染它。它与您单击表格行并让表格将行标记为选中无关(直接或可靠)。

换句话说,您不希望由于调用该getValueAt()方法而改变表模型的状态。

如果您的getValueAt()方法使用该行作为数组的索引,那将是最简单的。但是,看起来每个模块的行数是可变的。在这种情况下,如果您不能通过算法生成正确的索引,您可以在构造函数中“展平”一次数据,并简化查找代码。

也就是说,您的表模型包含:

private List<RowData> rowData

和一个包含行数据的内部类:

private static class RowData
{
  private Moddule;
  private Assignment assignment;

  ...
}

在你的构造函数中:

public AssignmentsTableModel(List<Module> modules)
{
  this.rowData = new ArrayList<Module>();

  for (Module module : modules)
  {
    for (Assignment assignment : module.getAssignments())
    {
      rowData.add(new RowData(module, assignment));
    }
  }
}

这简化了您的getValueAt()方法:

@Override
public Object getValueAt(int row, int column) {
    final RowData row = this.rowData.get(row);

    switch (column){
        case 0: return row.getModule().getIdentifier();
        case 1: return row.getModule().getTitle();
        case 2: return row.getAssignment().getTitle();
        case 3: return row.getAssignment().Author();
        case 4: return row.getAssignment().getSet();
        case 5: return row.getAssignment().getDue();
        case 6: return row.getAssignment().getWeighting();
        default: return null;
    }
}
于 2012-04-02T00:33:06.727 回答
1
  • 选择基本上与您没有关系TableModel,也许您未在此处发布的其余代码或方法中存在另一个问题,

  • 为了基本成功,我想念另一种方法来添加/修改/删除数据、行AbstractTableModel或列(反之亦然)ModelView

  • 使用DefaultTableModel,也许没有理由使用AbstractTableModel

  • 您可以比较基本AbstractTableModel所需的方法,直接在 API 中使用Object[][]或实现,而不是另一个 Array 或 Hash 的类型,Vector<Vector<Object>>

  • 请阅读教程JTable 的工作原理

编辑

@Dean 写道:

that I can't see another way around but when an update in the table happens

这就是为什么setXxxMethodsfireXxxXxx内部有方法)存在的原因,Oracle 教程JTable中提到的次数更多

于 2012-04-02T00:00:28.800 回答