Swing JTable,从 DefaultTableModel 中删除行时出现异常

Swing JTable, exception while removing rows from DefaultTableModel

我编写了以下函数,我可以使用它从 DefaultTableModel 中删除行:

private static void removeTableRows(JTable tableToClear){
    DefaultTableModel defaultModel = 
            (DefaultTableModel) tableToClear.getModel();
    int rows = defaultModel.getRowCount();

      while(rows>0) {
         defaultModel.removeRow(0);
      }
      String[] test = {"a","b","c"};
     defaultModel.addRow(test);
}

抛出以下异常:

Exception in thread "AWT-EventQueue-0" java.lang.ArrayIndexOutOfBoundsException: 0 >= 0
    at java.util.Vector.elementAt(Vector.java:427)
    at javax.swing.table.DefaultTableModel.getValueAt(DefaultTableModel.java:632)
    at org.swx.nursing.tools.gui.DefaultTableGui.tableChanged(DefaultTableGui.java:217)
    at javax.swing.table.AbstractTableModel.fireTableChanged(AbstractTableModel.java:280)
    at javax.swing.table.AbstractTableModel.fireTableRowsDeleted(AbstractTableModel.java:245)
    at javax.swing.table.DefaultTableModel.removeRow(DefaultTableModel.java:447)
    at org.swx.nursing.tools.gui.DefaultTableGui.removeTableRows(DefaultTableGui.java:357)
    at org.swx.nursing.tools.gui.DefaultTableGui.actionPerformed(DefaultTableGui.java:330)
    at javax.swing.AbstractButton.fireActionPerformed(AbstractButton.java:1995)
    at javax.swing.AbstractButton$Handler.actionPerformed(AbstractButton.java:2318)
    at javax.swing.DefaultButtonModel.fireActionPerformed(DefaultButtonModel.java:387)
    at javax.swing.DefaultButtonModel.setPressed(DefaultButtonModel.java:242)
    at javax.swing.plaf.basic.BasicButtonListener.mouseReleased(BasicButtonListener.java:236)
    at java.awt.Component.processMouseEvent(Component.java:6297)
    at javax.swing.JComponent.processMouseEvent(JComponent.java:3275)
    at java.awt.Component.processEvent(Component.java:6062)
    at java.awt.Container.processEvent(Container.java:2039)
    at java.awt.Component.dispatchEventImpl(Component.java:4660)
    at java.awt.Container.dispatchEventImpl(Container.java:2097)
    at java.awt.Component.dispatchEvent(Component.java:4488)
    at java.awt.LightweightDispatcher.retargetMouseEvent(Container.java:4575)
    at java.awt.LightweightDispatcher.processMouseEvent(Container.java:4236)
    at java.awt.LightweightDispatcher.dispatchEvent(Container.java:4166)
    at java.awt.Container.dispatchEventImpl(Container.java:2083)
    at java.awt.Window.dispatchEventImpl(Window.java:2489)
    at java.awt.Component.dispatchEvent(Component.java:4488)
    at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:674)
    at java.awt.EventQueue.access0(EventQueue.java:81)
    at java.awt.EventQueue.run(EventQueue.java:633)
    at java.awt.EventQueue.run(EventQueue.java:631)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.AccessControlContext.doIntersectionPrivilege(AccessControlContext.java:87)
    at java.security.AccessControlContext.doIntersectionPrivilege(AccessControlContext.java:98)
    at java.awt.EventQueue.run(EventQueue.java:647)
    at java.awt.EventQueue.run(EventQueue.java:645)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.AccessControlContext.doIntersectionPrivilege(AccessControlContext.java:87)
    at java.awt.EventQueue.dispatchEvent(EventQueue.java:644)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:269)
    at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:184)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:174)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:169)
    at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:161)
    at java.awt.EventDispatchThread.run(EventDispatchThread.java:122)

tablechanged方法重写如下:

 //Add table cell listener
    model.addTableModelListener(new TableModelListener(){

        public void tableChanged(TableModelEvent e) {
             int row = e.getFirstRow();
             int col = e.getColumn();
             TableModel tableModel = (TableModel)e.getSource();
             Object data = null ;

             try{
                 data = tableModel.getValueAt(row, col);
             } catch (Exception ex) {
                 LOGGER.error("Failed to access table data: "+ex.getMessage());
             }

             boolean ishotKeyColEmpty = tableModel.getValueAt(row, 0).toString().isEmpty();
             boolean ishTypeColEmpty = tableModel.getValueAt(row, 2).toString().isEmpty();     
             /*
              * If the Application path column is empty, then clear the contents from
              * the hotKey and Type cells for this row 
              * 
              */
             if(data != null 
                     && data.toString().equals("") 
                     && !ishotKeyColEmpty 
                     && !ishTypeColEmpty) {
                 tableModel.setValueAt("", row, 0);
                 tableModel.setValueAt("", row, 2);
             }      
        }
    });

我不确定是什么导致了这个异常..

您的代码中有 2 个问题。

1) 尝试从 none 现有的 row/column 获取值 - 在 modellistner.

2)即使没有更多的行在 removeTableRows 方法

中,也不停止地连续删除行

让我先解释第二个错误。

2) 在 while 循环中你没有更新行 count.you 正在获取行数但是这个 while 循环条件永远不会变成 false.for 例如你有 3 行并且计数现在是 3在 while 循环中你正在检查 (rows > 0) 无论它循环了多少次它总是正确的所以当循环运行第 4 次时你会得到一个错误因为没有更多的行要删除。你必须重新计算行数

将方法更改为

DefaultTableModel defaultModel= (DefaultTableModel) tableToClear.getModel();
int rows = defaultModel.getRowCount();
while (rows > 0) {
       defaultModel.removeRow(0);
       rows = defaultModel.getRowCount(); // this is very important
}
String[] test = {"a", "b", "c"};
defaultModel.addRow(test);


注意:

您可以使用方法 defaultModel.RowCount(0); 方法来删​​除所有行。您不需要使用循环来完成此操作

1) 确保 row 和 col 为 0 或大于它。因为它们可以是 -1,而它们不是 exist.then 你会得到很多错误

所以这一行很重要

 (row >= 0 && col >= 0){}

还有这里

boolean ishotKeyColEmpty = tableModel.getValueAt(row, 0).toString().isEmpty();

您必须在调用前安全地检查 getValueAt(row, 0) 是否为空 toString().isEmpty() 方法。 在你的

完整代码

addTableModelListener(new TableModelListener() {

            @Override
            public void tableChanged(TableModelEvent e) {

                int row = e.getFirstRow();
                int col = e.getColumn();
                TableModel tableModel = (TableModel) e.getSource();
                Object data = null;

                try {
                    if (row >= 0 && col >= 0) {
                        data = tableModel.getValueAt(row, col);

                        boolean ishotKeyColEmpty = false;
                        boolean ishTypeColEmpty = false;
                        if (tableModel.getValueAt(row, 0) != null) {
                            ishotKeyColEmpty = tableModel.getValueAt(row, 0).toString().isEmpty();
                        }
                        if (tableModel.getValueAt(row, 2) != null) {
                            ishTypeColEmpty = tableModel.getValueAt(row, 2).toString().isEmpty();
                        }

                        if (data != null && data.toString().equals("") && !ishotKeyColEmpty && !ishTypeColEmpty) {
                            tableModel.setValueAt("", row, 0);
                            tableModel.setValueAt("", row, 2);
                        }
                    }
                } catch (Exception ex) {
                    ex.printStackTrace();
                }

            }

        }