JTable如何在插入、删除或更新数据后刷新表格模型。

95

这是我的jTable

private JTable getJTable() {
    String[] colName = { "Name", "Email", "Contact No. 1", "Contact No. 2",
            "Group", "" };
    if (jTable == null) {
        jTable = new JTable() {
            public boolean isCellEditable(int nRow, int nCol) {
                return false;
            }
        };
    }
    DefaultTableModel contactTableModel = (DefaultTableModel) jTable
            .getModel();
    contactTableModel.setColumnIdentifiers(colName);
    jTable.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
    return jTable;
}

我将调用这个方法从数据库中检索数据并将其放入表格模型中。

public void setUpTableData() {
    DefaultTableModel tableModel = (DefaultTableModel) jTable.getModel();
    ArrayList<Contact> list = new ArrayList<Contact>();
    if (!con.equals(""))
        list = sql.getContactListsByGroup(con);
    else
        list = sql.getContactLists();
    for (int i = 0; i < list.size(); i++) {
        String[] data = new String[7];

            data[0] = list.get(i).getName();
            data[1] = list.get(i).getEmail();
            data[2] = list.get(i).getPhone1();
            data[3] = list.get(i).getPhone2();
            data[4] = list.get(i).getGroup();
            data[5] = list.get(i).getId();

        tableModel.addRow(data);
    }
    jTable.setModel(tableModel);
}

目前我正在使用这种方法来更新表格数据后刷新表格。 我将首先清除表格。

DefaultTableModel tableModel = (DefaultTableModel) jTable.getModel();
tableModel.setRowCount(0);

然后重新构建表格模型,以便刷新jTable。但我在想是否有任何最佳实践或更好的方法来做到这一点?


非常老的问题,没有足够的细节来确定问题的确切原因(DefaultTableModel根据需要触发通知,因此错误可能是其使用方式,最有可能的)- 仍然令当前读者感到困惑 ;) - kleopatra
6个回答

132
如果您希望通知您的JTable数据发生更改,请使用 tableModel.fireTableDataChanged()
根据文档
通知所有监听器表中所有单元格值可能已更改。行数也可能已更改,JTable应该从头重新绘制表格。表格的结构(如列的顺序)被认为是相同的。

你的意思是我每次更新时都要调用tableModel.fireTableDataChanged()吗? - user236501
3
如果你只是插入新行,可以使用 fireTableRowsInserted 方法。另一方面,DefaultTableModel.addRow 的实现应该已经处理了这个问题... 你的表格难道不会被 addRow 刷新吗? - Peter Lang
4
使用 setValueAt 方法时,DefaultTableModel 会触发事件。 - Peter Lang
23
需要翻译的内容:One should notice though that the call must be made on casted javax.swing.table.AbstractTableModel, since interface TableModel does not have the method mentioned above。然而,需要注意的是,必须在转换为javax.swing.table.AbstractTableModel后进行调用,因为接口TableModel没有上述提到的方法。 - Milan Aleksić
技术上是正确的,但是a)在DefaultTableModel中不需要(它的修改方法会触发适当的通知),b)让客户端代码触发通知在概念上是错误的 - 这是模型的固有责任。因此,在使用模型时存在问题,不幸的是这个问题在问题中没有显示。 - kleopatra

22

对于你的情况,更快的方式是:

    jTable.repaint(); // Repaint all the component (all Cells).

当一个或少数个单元格发生变化时,最优化的方法是:

    ((AbstractTableModel) jTable.getModel()).fireTableCellUpdated(x, 0); // Repaint one cell.

1
我实际上发现 jTable.invalidate() 是一个真正强制重绘的方法。 - Kevin Qiu
正确,但验证方法是布局过程的一部分,并影响容器的父级。因此,如果您需要重新布局,请使用它。 http://docs.oracle.com/javase/7/docs/api/java/awt/Container.html#invalidate() - Daniel De León

8

试一下

public void setUpTableData() {
    DefaultTableModel tableModel = (DefaultTableModel) jTable.getModel();

    /**
    * additional code.
    **/
    tableModel.setRowCount(0);
    /**/
    ArrayList<Contact> list = new ArrayList<Contact>();
    if (!con.equals(""))
        list = sql.getContactListsByGroup(con);
    else
        list = sql.getContactLists();
    for (int i = 0; i < list.size(); i++) {
        String[] data = new String[7];

        data[0] = list.get(i).getName();
        data[1] = list.get(i).getEmail();
        data[2] = list.get(i).getPhone1();
        data[3] = list.get(i).getPhone2();
        data[4] = list.get(i).getGroup();
        data[5] = list.get(i).getId();

        tableModel.addRow(data);
    }
    jTable.setModel(tableModel);
    /**
    * additional code.
    **/
    tableModel.fireTableDataChanged();
    /**/
}

2
你不需要在最后使用jTable.setModel(tableModel),因为你已经在开始时获取了表格的模型。 - David George

1

使用 java.util.Observablejava.util.Observer 不是更好吗?这样会导致表格更新。


7
不确定,不要这样做,为什么模拟JTable的内置选项并从EDT中移出。 - mKorbel

0
To inform the Jtable about data change do this - 

DefaultTableModel tableModel = new DefaultTableModel(yourData, yourColumns);
//Set data and columns in tableModel to JTable
JTable yourTable= new JTable(model);

//After data Changed fire change detector
tableModel.fireTableDataChanged();

-4

我在我的JTable中这样做,它会在300毫秒后自动刷新;

DefaultTableModel tableModel = new DefaultTableModel(){
public boolean isCellEditable(int nRow, int nCol) {
                return false;
            }
};
JTable table = new JTable();

Timer t = new Timer(300, new ActionListener() {

            @Override
            public void actionPerformed(ActionEvent e) {
                addColumns();
                remakeData(set);
                table.setModel(model);
            }
        });
        t.start();

private void addColumns() {
        model.setColumnCount(0);
        model.addColumn("NAME");
            model.addColumn("EMAIL");} 

 private void remakeData(CollectionType< Objects > name) {
    model.setRowCount(0);
    for (CollectionType Objects : name){
    String n = Object.getName();
    String e = Object.getEmail();
    model.insertRow(model.getRowCount(),new Object[] { n,e });
    }}

我怀疑它在处理超过500个对象时是否有效,唯一的解决方法是在你的类中实现TableModelListener,但我不太明白如何使用它。请参考http://download.oracle.com/javase/tutorial/uiswing/components/table.html#modelchange


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接