Popolare JTable con un numero elevato di righe

Mi piacerebbe popolare un JTable durante il runtime con molte righe (diciamo 10000). Ma tutti i miei tentativi sono molto poveri e inefficienti.

Il punto di partenza è il metodo addData che ottiene una lista di oggetti che rappresentano una riga. Ho provato a riempire il tavolo tramite SwingWorker ma questo funziona solo per i piccoli dati per me.

Un altro tentativo è stato quello di impostare i dati direttamente senza utilizzare alcun tipo di thread, ma anche questo è molto lento, almeno l’interfaccia utente non è bloccata come nel caso di SwingWorker.

Quindi, come lo fai è generale? La tabella deve essere riempita riga per riga o in senso orario, ma non tutti per uno e la barra di scorrimento verticale dovrebbe essere scorrevole nel frattempo.

My TableModel:

 public class MyTableModel extends AbstractTableModel { /** * */ private static final long serialVersionUID = 1L; String[] columnNames; public Map data = new LinkedHashMap(); public MyTableModel(String[] header) { columnNames = header; } public String getColumnName(int col) { return columnNames[col].toString(); } @Override public int getColumnCount() { return columnNames.length; } @Override public int getRowCount() { return data.size(); } @Override public Object getValueAt(int row, int col) { . . return value; } public void addRow(long id, MyDataObject o) { data.put(id, m); fireTableRowsInserted(0,nqm_messages.size()-1); } } 

Implementazione di SwingWorker:

 class TableSwingWorker extends SwingWorker { private final MyTableModel tableModel; List messages; public TableSwingWorker(MyTableModel tableModel, List dataList) { this.tableModel = tableModel; this.messages = new LinkedList(mm); } @Override protected MyTableModel doInBackground() throws Exception { for(MyDataObject s : messages) { publish(s); } return tableModel; } @Override protected void process(List chunks) { for(MyDataObject row : chunks){ Long l = Long.parseLong(row.getId()); tableModel.addRow(l, row); } } } 

Aggiungi oggetti a JTable:

 public void addData(List o) { MyTableModel m = (MyTableModel)table.getModel(); (new TableSwingWorker(m,o)).execute(); //for(int i=0; i < mm.size();i++) { // long l = Long.parseLong(mm.get(i).getId()); // m.addRow(l, mm.get(i)); //} } 

Quindi, una serie di cose sono state identificate dai commenti …

  • È necessario triggersre correttamente il metodo della riga inserita, indicando solo le righe che sono state aggiunte e in cui sono state aggiornate. Questo è molto importante, poiché il tavolo è stato ottimizzato per la velocità
  • Dovresti fornire un metodo di aggiunta in batch per il tuo modello di tabella, che ti consente di aggiungere più righe in modo più semplice in una sola o più fasi possibili
  • Dovresti fare in modo che SwingWorker dorma o ceda periodicamente, in modo da avere il tempo di pubblicare i risultati.

Quindi, in questo esempio, aggiungo 1, 000, 000 righe. Nel mio test ci sono voluti poco meno di 1 secondo …

 import java.awt.BorderLayout; import java.awt.Component; import java.awt.EventQueue; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Arrays; import java.util.Date; import java.util.List; import javax.swing.JFrame; import javax.swing.JScrollPane; import javax.swing.JTable; import javax.swing.SwingWorker; import javax.swing.UIManager; import javax.swing.UnsupportedLookAndFeelException; import javax.swing.table.AbstractTableModel; import javax.swing.table.DefaultTableCellRenderer; public class TestTableLoad01 { public static void main(String[] args) { new TestTableLoad01(); } public TestTableLoad01() { EventQueue.invokeLater(new Runnable() { @Override public void run() { try { UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName()); } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) { } MyTableModel model = new MyTableModel(); JTable table = new JTable(model); table.setDefaultRenderer(Date.class, new TimeCellRenderer()); JFrame frame = new JFrame("Testing"); frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); frame.setLayout(new BorderLayout()); frame.add(new JScrollPane(table)); frame.pack(); frame.setLocationRelativeTo(null); frame.setVisible(true); TableSwingWorker worker = new TableSwingWorker(model); worker.execute(); } }); } public class TimeCellRenderer extends DefaultTableCellRenderer { private DateFormat df; public TimeCellRenderer() { df = new SimpleDateFormat("HH:mm:ss"); } @Override public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) { if (value instanceof Date) { value = df.format(value); } super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column); return this; } } public class MyTableModel extends AbstractTableModel { private String[] columnNames = new String[]{"Date", "Row"}; private List data; public MyTableModel() { data = new ArrayList<>(25); } @Override public Class< ?> getColumnClass(int columnIndex) { return columnIndex == 0 ? Date.class : Integer.class; } @Override public String getColumnName(int col) { return columnNames[col]; } @Override public int getColumnCount() { return columnNames.length; } @Override public int getRowCount() { return data.size(); } @Override public Object getValueAt(int row, int col) { RowData value = data.get(row); return col == 0 ? value.getDate() : value.getRow(); } public void addRow(RowData value) { int rowCount = getRowCount(); data.add(value); fireTableRowsInserted(rowCount, rowCount); } public void addRows(RowData... value) { addRows(Arrays.asList(value)); } private void addRows(List rows) { int rowCount = getRowCount(); data.addAll(rows); fireTableRowsInserted(rowCount, getRowCount() - 1); } } public class RowData { private Date date; private int row; public RowData(int row) { this.date = new Date(); this.row = row; } public Date getDate() { return date; } public int getRow() { return row; } } public class TableSwingWorker extends SwingWorker { private final MyTableModel tableModel; public TableSwingWorker(MyTableModel tableModel) { this.tableModel = tableModel; } @Override protected MyTableModel doInBackground() throws Exception { // This is a deliberate pause to allow the UI time to render Thread.sleep(2000); System.out.println("Start polulating"); for (int index = 0; index < 1000000; index++) { RowData data = new RowData(index); publish(data); Thread.yield(); } return tableModel; } @Override protected void process(List chunks) { System.out.println("Adding " + chunks.size() + " rows"); tableModel.addRows(chunks); } } }