在 Jtable 中对行进行排序时出错



我正在尝试在我的 Jtable 上设置RowSorter,我使用 setAutoCreateRowSorter(Boolean b) 方法对行进行排序

table.setAutoCreateRowSorter(true);

但是当我把桌子做成rawSorted时,我得到了一个奇怪的错误!

当我想删除一行时,冲突是可见的,我使用了fireTableRowsDeleted().

int raw = table.getSelectedRow();  // the index of raw that i want to delete it
System.out.println(raw);
model.delte_raw(raw); // model is my table model
public void delte_raw(int raw)
         {
             if (!ls.isEmpty()) {
         this.fireTableRowsDeleted(raw+1, raw);
         ls.remove(raw);
         }

我想向您展示在 2 种情况下返回上述代码的结果是什么:

案例1:

当我将我的表格设置为未原始排序时:

table.setAutoCreateRowSorter(false);

当我删除一行时,一切都成功了。

案例2:

当我将我的表格设置为原始排序时:

table.setAutoCreateRowSorter(true);

当我删除一行时,出现如下错误:

Exception in thread "AWT-EventQueue-0" java.lang.IndexOutOfBoundsException: Invalid range
    at javax.swing.DefaultRowSorter.checkAgainstModel(DefaultRowSorter.java:921)
    at javax.swing.DefaultRowSorter.rowsDeleted(DefaultRowSorter.java:878)
    at javax.swing.JTable.notifySorter(JTable.java:4277)
    at javax.swing.JTable.sortedTableChanged(JTable.java:4121)
    at javax.swing.JTable.tableChanged(JTable.java:4398)
    at javax.swing.table.AbstractTableModel.fireTableChanged(AbstractTableModel.java:296)
    at javax.swing.table.AbstractTableModel.fireTableRowsDeleted(AbstractTableModel.java:261)

我认为错误出在我的默认行排序器中,所以我定义我的特定cellRenderer如下:

//    final TableCellRenderer r = table.getTableHeader().getDefaultRenderer();
//TableCellRenderer wrapper = new TableCellRenderer() {
//            private Icon ascendingIcon = new ImageIcon("images/e.png");
//            private Icon descendingIcon = new ImageIcon("images/e.png");
//
//    @Override
//    public Component getTableCellRendererComponent(JTable table,
//            Object value, boolean isSelected, boolean hasFocus,
//            int row, int column) 
//    {
//        Component comp = r.getTableCellRendererComponent(table, value, isSelected, 
//            hasFocus, row, column);
//        if (comp instanceof JLabel) {
//            JLabel label = (JLabel) comp;
//            label.setIcon(getSortIcon(table, column));
//        }
//        return comp;
//    }
//
//    /**
//     * Implements the logic to choose the appropriate icon.
//     */
//    private Icon getSortIcon(JTable table, int column) {
//        SortOrder sortOrder = getColumnSortOrder(table, column);
//        if (SortOrder.UNSORTED == sortOrder) {
//            return null;
//        }
//        return SortOrder.ASCENDING == sortOrder ? ascendingIcon : descendingIcon;
//    }
//
//    private SortOrder getColumnSortOrder(JTable table, int column) {
//        if (table == null || table.getRowSorter() == null) {
//            return SortOrder.UNSORTED;
//        }
//        List<? extends RowSorter.SortKey> keys = table.getRowSorter().getSortKeys();
//        if (keys.size() > 0) {
//                    RowSorter.SortKey key = keys.get(0);
//            if (key.getColumn() == table.convertColumnIndexToModel(column)) {
//                return key.getSortOrder();
//            }
//        }
//        return SortOrder.UNSORTED;
//    }
//
//};
//table.getTableHeader().setDefaultRenderer(wrapper);

但同样,同样的错误!

为什么我会收到此错误?我用谷歌搜索了很多,但要么我使用了错误的关键字,要么互联网上没有简单的解决方案。

在表模型中:

public void delte_raw(int raw) {
    if (!ls.isEmpty()) {
        this.fireTableRowsDeleted(raw+1, raw); // why raw+1 ???
        ls.remove(raw);
    }
}

当你的表模型从AbstractTableModel扩展并查看fireTableRowsDeleted(int firstRow, int lastRow) javadoc:

通知所有侦听器在 [firstRow, lastRow] 、 包含,已被删除。

所以它应该是:

public void delte_raw(int raw) {
    if (!ls.isEmpty()) {            
        ls.remove(raw); // remove the row index from the List and then fire the event
        fireTableRowsDeleted(raw, raw);
    }
}

了解异常来源:查看DefaultRowSorter.checkAgainstModel(int firstRow, int endRow)实现:

private void checkAgainstModel(int firstRow, int endRow) {
    if (firstRow > endRow || firstRow < 0 || endRow < 0 ||
            firstRow > modelRowCount) {
        throw new IndexOutOfBoundsException("Invalid range");
    }
}

如您所见,使用[raw+1,raw]范围调用此方法会导致IndexOutOfBoundsException

编辑

正如@mKorbel巧妙地指出的那样,我完全忽略了这一点:

int raw = table.getSelectedRow();  // this is the index in the view
model.delte_raw(raw); // convert raw in the right model index is needed

您需要在正确的模型索引中转换raw。否则,它可能会导致副作用,因为在排序表中,视图中的选定索引很可能与其相关模型的索引不同:

int raw = table.getSelectedRow();  // this is the index in the view
model.delte_raw(table.convertRowIndexToModel(raw)); // perfect

参见 JTable.convertRowIndexToModel(int viewRowIndex)

最新更新