JTable los valores duplicados en fila

Tengo una JTable llena con un DataModel personalizado (pegado a continuación) y cuando llamo al método populate() , aparece llenar la tabla con datos duplicados: cada fila se llena con el mismo valor una y otra vez. Sin embargo, en una inspección más cercana (simplemente imprimiendo () ing el campo ‘datos’), el modelo de datos no tiene la culpa: contiene los datos correctos, en el formato que espero. ¿Lo que da?

 import java.util.ArrayList; import javax.swing.table.AbstractTableModel; @SuppressWarnings("serial") // we don't expect this app to ever use serialized classes. EVER. public class CollectionDataModel extends AbstractTableModel { private ArrayList<ArrayList> data; public CollectionDataModel() { data = new ArrayList<ArrayList>(); } @Override public int getColumnCount() { if(data.isEmpty()) return 0; return data.get(0).size(); } @Override public int getRowCount() { return data.size(); } @Override public Object getValueAt(int rowIndex, int columnIndex) { if(rowIndex > getRowCount()) return null; if(columnIndex > getColumnCount()) return null; return data.get(rowIndex).get(columnIndex); } public void populate(Collection c) { data.clear(); for(Item i : c.getItems()) { ArrayList row = new ArrayList(); for(Property p : i.getProperties().values()) { row.add(p.toString()); } data.add(row); } fireTableDataChanged(); } } 

Aquí hay un ejemplo completo que puede ser útil. Como el Map muestra no se puede modificar, le remito al ejemplo de @ mKorbel sobre cómo anular isCellEditable() y setValueAt() .

 import java.awt.EventQueue; import java.awt.GridLayout; import java.util.Map; import javax.swing.JFrame; import javax.swing.JPanel; import javax.swing.JScrollPane; import javax.swing.JTable; import javax.swing.table.AbstractTableModel; /** @see https://stackoverflow.com/questions/9132987 */ public class EnvTableTest extends JPanel { public EnvTableTest() { this.setLayout(new GridLayout()); this.add(new JScrollPane(new JTable(new EnvDataModel()))); } private static class EnvDataModel extends AbstractTableModel { private Map data = System.getenv(); private String[] keys; public EnvDataModel() { keys = data.keySet().toArray(new String[data.size()]); } @Override public String getColumnName(int col) { if (col == 0) { return "Key"; } else { return "Value"; } } @Override public int getColumnCount() { return 2; } @Override public int getRowCount() { return data.size(); } @Override public Object getValueAt(int row, int col) { if (col == 0) { return keys[row]; } else { return data.get(keys[row]); } } } private void display() { JFrame f = new JFrame("EnvTableTest"); f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); f.add(this); f.pack(); f.setLocationRelativeTo(null); f.setVisible(true); } public static void main(String[] args) { EventQueue.invokeLater(new Runnable() { @Override public void run() { new EnvTableTest().display(); } }); } } 

Podría tratar de hacer que los cambios de población sean más atómicos.

 public void populate(Collection c) { ArrayList> data2 = new ArrayList>(); for(Item i : c.getItems()) { ArrayList row = new ArrayList(); for(Property p : i.getProperties().values()) { row.add(p.toString()); } data2.add(row); } data = data2; fireTableDataChanged(); } 

Supongo que se volverá a llamar a populate antes de que se populate llamada anterior. Y probablemente c se cambie durante su iteración.

1) su TableModel está un_completed, echo de menos allí montones o métodos necesarios para life_cycle JTable's , comenzando con TableHeader etc.

2) dado que hay muchos modelos AbstactTableModels basados ​​en HashMap, sugiero devolver el tipo de arrays implementado en API directamente

 Vector> data; String[][] or Object[][] 

en lugar de

 ArrayList> data; 

explicaciones simples es que XxxList vuelve a colocar la columna y Vector o String [] devuelve Fila

3) Sugeriría usar DefaultTableModel directamente, entonces nunca necesitarás resolver duplicados o columnas / filas perdidas