JTable Як оновити модель таблиці після вставки, видалення або оновлення даних.


89

Це моя 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. Але я думав, чи є якісь найкращі практики чи кращий спосіб це зробити?

Відповіді:


119

Якщо ви хочете повідомити вас JTableпро зміни ваших даних, використовуйте
tableModel.fireTableDataChanged()

З документації :

Повідомляє всіх слухачів про те, що всі значення комірок у рядках таблиці, можливо, змінилися. Кількість рядків також може змінитися, і JTable повинен перекроювати таблицю з нуля. Структура таблиці (як у порядку стовпців) вважається однаковою.


ви маєте на увазі, що я закликаю цей tableModel.fireTableDataChanged () щоразу, коли я роблю оновлення?
user236501

3
@ newbie123: Якщо ви вставляєте лише нові рядки, замість цього ви можете використовувати fireTableRowsInserted . З іншого боку, реалізація DefaultTableModel.addRowвже повинна це обробляти ... Чи не оновлюється ваш стіл addRow?
Peter Lang,

4
Використовуйте setValueAt, DefaultTableModelспрацьовує подію.
Пітер Ленг,

18
Хоча слід зауважити, що дзвінок повинен здійснюватися на кастовому рівні javax.swing.table.AbstractTableModel, оскільки в інтерфейсі TableModelнемає методу, згаданого вище
Milan Aleksić

@PeterLang см моє запитання ласка: stackoverflow.com/questions/18282753 / ...
Саяд

20

Швидший спосіб для вашої справи:

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

Оптимізований спосіб зміни однієї або декількох комірок:

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

1
Я насправді виявив, що jTable.invalidate () - це метод, який фактично змусив перемалювати.
Kevin Qiu

Правильно, але метод перевірки є частиною процесу компонування та впливає на батьків контейнера. Отже, якщо вам потрібно переробити макет, використовуйте його. docs.oracle.com/javase/7/docs/api/java/awt/…
Даніель Де Леон,

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) в кінці, оскільки ви вже отримали модель таблиці на початку.
Девід Джордж

2
DefaultTableModel dm = (DefaultTableModel)table.getModel();
dm.fireTableDataChanged(); // notifies the JTable that the model has changed

3
ні, ні ... DefaultTableModel реалізував цю подію і застосував правильно ...
mKorbel

1

Чи не було б краще використовувати, java.util.Observableі java.util.Observerце призведе до оновлення таблиці?


6
ні, звичайно ні, не робіть цього, чому імітується вбудована опція
JTable

-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

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.