【问题标题】:JTable How to refresh table model after insert delete or update the data.JTable 如何在插入删除或更新数据后刷新表模型。
【发布时间】:2011-03-11 21:13:21
【问题描述】:

这是我的 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。但我在想是否有任何最佳实践或更好的方法来做到这一点?

【问题讨论】:

    标签: java swing refresh jtable tablemodel


    【解决方案1】:

    如果您想将您的数据更改通知您的JTable,请使用
    tableModel.fireTableDataChanged()

    来自documentation

    通知所有侦听器表格行中的所有单元格值可能已更改。行数也可能发生了变化,JTable 应该从头开始重新绘制表格。假设表的结构(如列的顺序)是相同的。

    【讨论】:

    • 你的意思是我每次更新时都会调用这个 tableModel.fireTableDataChanged() 吗?
    • @newbie123:如果只插入新行,可以改用fireTableRowsInserted。另一方面,DefaultTableModel.addRow 的实现应该已经处理了……你的表不是被addRow 刷新了吗?
    • 使用setValueAt, DefaultTableModel 触发事件。
    • 大家应该注意到,调用必须在 javax.swing.table.AbstractTableModel 上进行,因为接口 TableModel 没有上面提到的方法
    【解决方案2】:

    您的情况更快的方法是:

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

    一个或几个cell变化时的优化方式:

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

    【讨论】:

    • 我实际上发现 jTable.invalidate() 是实际上强制重绘的方法。
    • 对,但是 validate 方法是布局过程的一部分,也会影响容器的父级。因此,如果您需要重做布局,请使用它。 docs.oracle.com/javase/7/docs/api/java/awt/…
    【解决方案3】:

    试试这个

    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();
        /**/
    }
    

    【讨论】:

    • 你不需要 jTable.setModel(tableModel) 最后因为你已经在开始时得到了表格的模型。
    【解决方案4】:
    DefaultTableModel dm = (DefaultTableModel)table.getModel();
    dm.fireTableDataChanged(); // notifies the JTable that the model has changed
    

    【讨论】:

    • 不,不是... DefaultTableModel 已经实现了这个事件,并且正确实现了...
    【解决方案5】:

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

    【讨论】:

    • 不,当然不,不要那样做,为什么模拟JTable的内置选项并从EDT退出
    【解决方案6】:

    我在我的 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

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2013-09-22
      • 1970-01-01
      • 1970-01-01
      • 2014-09-06
      • 2015-02-02
      • 2014-08-11
      • 2012-12-31
      • 1970-01-01
      相关资源
      最近更新 更多