JTable Jak odświeżyć model tabeli po wstawieniu, usuń lub zaktualizuj dane.

89

To jest mój 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;
}

Wywołam tę metodę, aby pobrać dane z bazy danych i umieścić je w modelu tabeli

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);
}

Obecnie używałem tej metody do odświeżania tabeli po zaktualizowaniu danych tabeli. Najpierw posprzątam ze stołu

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

a następnie ponownie zrestrukturyzuj model tabeli, aby odświeżył jTable. Ale zastanawiałem się, czy są jakieś najlepsze praktyki lub lepszy sposób, aby to zrobić?

user236501
źródło

Odpowiedzi:

119

Jeśli chcesz powiadomić Cię JTableo zmianie Twoich danych, użyj
tableModel.fireTableDataChanged()

Z dokumentacji :

Powiadamia wszystkich odbiorników, że wszystkie wartości komórek w wierszach tabeli mogły ulec zmianie. Liczba wierszy również mogła się zmienić i JTable powinien przerysować tabelę od zera. Zakłada się, że struktura tabeli (zgodnie z kolejnością kolumn) jest taka sama.

Peter Lang
źródło
czy masz na myśli, że nazywam tę tableModel.fireTableDataChanged () za każdym razem, gdy robię aktualizację?
user236501
3
@ newbie123: Jeśli wstawisz tylko nowe wiersze, możesz zamiast tego użyć fireTableRowsInserted . Z drugiej strony implementacja DefaultTableModel.addRowpowinna już obsłużyć to ... Czy Twój stół nie jest odświeżany przez addRow?
Peter Lang,
4
Użyj setValueAt, DefaultTableModelodpala wydarzenie.
Peter Lang,
18
Należy jednak zauważyć, że połączenie musi być wykonane na castingu javax.swing.table.AbstractTableModel, ponieważ interfejs TableModelnie obsługuje wyżej wymienionej metody
Milan Aleksić
@PeterLang zobacz moje pytanie, proszę: stackoverflow.com/questions/18282753/…
Sajad
20

Szybszy sposób na rozwiązanie sprawy to:

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

Zoptymalizowany sposób w przypadku zmiany jednej lub kilku komórek:

    ((AbstractTableModel) jTable.getModel()).fireTableCellUpdated(x, 0); // Repaint one cell.
Daniel De León
źródło
1
Właściwie odkryłem, że metoda jTable.invalidate () wymusiła przerysowanie.
Kevin Qiu
Zgadza się, ale metoda walidacji jest częścią procesu układu i wpływa również na elementy nadrzędne kontenera. Więc jeśli potrzebujesz przerobić układ, użyj go. docs.oracle.com/javase/7/docs/api/java/awt/…
Daniel De León,
7

Spróbuj tego

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();
    /**/
}
Achille
źródło
2
Nie potrzebujesz jTable.setModel (tableModel) na końcu, ponieważ masz już model Table na początku.
David George,
2
DefaultTableModel dm = (DefaultTableModel)table.getModel();
dm.fireTableDataChanged(); // notifies the JTable that the model has changed
Sumit Singh
źródło
3
Nie, nie ... DefaultTableModel wdrożył tego zdarzenia i realizowane prawidłowo ...
mKorbel
1

Czy nie byłoby lepiej w użyciu java.util.Observablei java.util.Observerto spowodowałoby aktualizację tabeli?

Tomek
źródło
6
nie, na pewno nie, nie rób tego, dlaczego symulowałem wbudowaną opcję JTable i wychodzę z EDT
mKorbel
-4

Zrobiłem to w ten sposób w moim Jtable, jego automatyczne odświeżanie po 300 ms;

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 });
    }}

Wątpię, aby poradził sobie z dużą liczbą obiektów, takich jak ponad 500, innym sposobem jest zaimplementowanie TableModelListener w twojej klasie, ale nie rozumiałem, jak dobrze go używać. spójrz na http://download.oracle.com/javase/tutorial/uiswing/components/table.html#modelchange

nkvnkv
źródło