Java Can';t更新JTable';s信息
我使用ArrayList中包含的数据创建JTable,然后将JTable添加到JScrollPane中Java Can';t更新JTable';s信息,java,swing,arraylist,jtable,Java,Swing,Arraylist,Jtable,我使用ArrayList中包含的数据创建JTable,然后将JTable添加到JScrollPane中 ArrayList<Product> stock = new ArrayList<Product>(s); String[] col = {"Nombre", "Cantidad", "Descripci\u00F3n", "Contenido"}; DefaultTableModel tableModel = new DefaultTable
ArrayList<Product> stock = new ArrayList<Product>(s);
String[] col = {"Nombre", "Cantidad", "Descripci\u00F3n", "Contenido"};
DefaultTableModel tableModel = new DefaultTableModel(col,0);
JTable table = new JTable(tableModel);
Collections.sort(stock, new Comparator<Product>() {
public int compare(Product p1, Product p2) {
return p1.getNombre().compareTo(p2.getNombre());
}
});
for (int i = 0; i < stock.size(); i++){
String nombre = stock.get(i).getNombre();
String cantidad = stock.get(i).getCantidad();
String descripcion = stock.get(i).getDescripcion();
String contenido = stock.get(i).getContenido();
Object[] data = {nombre, cantidad, descripcion, contenido};
tableModel.addRow(data);
}
table.getColumnModel().getColumn(0).setPreferredWidth(width*3/18);
table.getColumnModel().getColumn(1).setPreferredWidth(width/9);
table.getColumnModel().getColumn(2).setPreferredWidth(width*8/18);
table.getColumnModel().getColumn(3).setPreferredWidth(width/9);
frame.getContentPane().setLayout(new BorderLayout(0, 0));
table.getTableHeader().setFont(new Font("Arial", Font.BOLD, f));
table.setFont(new Font("Arial", Font.PLAIN, f));
table.setBounds(1, 1, width*8/9, height);
table.setRowHeight(height/30);
table.setEnabled(false);
DefaultTableCellRenderer centerRenderer = new DefaultTableCellRenderer();
centerRenderer.setHorizontalAlignment( JLabel.CENTER );
for(int i = 0; i<4; i++){
table.getColumnModel().getColumn(i).setCellRenderer( centerRenderer);
}
JScrollPane sp = new JScrollPane(table);
frame.getContentPane().add(sp, BorderLayout.CENTER);
问题是JTable不会更新,但信息在ArrayList中确实会更改。JTable的数据不保存在ArrayList中,而是保存在表模型本身中。一句话:不要更新ArrayList,而是更新表模型本身,这里是DefaultTableModel,然后您的JTable应该显示新数据。幸运的是,这应该很容易做到,因为表模型有一些方法,允许您使用
getValueAt(…)
提取数据,并使用setValueAt(…)
更新其单元格中的值,如果需要添加新行,还可以使用addRow(…)
请注意,如果您通过DefaultTableModel进行更改,则无需直接调用fireTableDataChanged()
,事实上,您不应该调用此方法——这是模型的责任。同样,不需要revalidate()
或repaint()
您的JTable
有关血淋淋的细节,请查看
另一方面,如果您绝对需要使用ArrayList作为JTable的数据核心,那么您不应该使用DefaultTableModel,而是通过扩展AbstractTableModel并使用ArrayList作为其数据核心来创建自己的TableModel。如果您这样做,那么您的模型的代码应该注意在更改模型的状态后调用相应的
fire…(…)
方法。JTable的数据不保存在ArrayList中,而是驻留在表模型本身中。一句话:不要更新ArrayList,而是更新表模型本身,这里是DefaultTableModel,然后您的JTable应该显示新数据。幸运的是,这应该很容易做到,因为表模型有一些方法,允许您使用getValueAt(…)
提取数据,并使用setValueAt(…)
更新其单元格中的值,如果需要添加新行,还可以使用addRow(…)
请注意,如果您通过DefaultTableModel进行更改,则无需直接调用fireTableDataChanged()
,事实上,您不应该调用此方法——这是模型的责任。同样,不需要revalidate()
或repaint()
您的JTable
有关血淋淋的细节,请查看
另一方面,如果您绝对需要使用ArrayList作为JTable的数据核心,那么您不应该使用DefaultTableModel,而是通过扩展AbstractTableModel并使用ArrayList作为其数据核心来创建自己的TableModel。如果您这样做,那么您的模型的代码应该注意在更改模型的状态后调用相应的
fire…(…)
方法。谢谢,我确实必须更改ArrayList中的值,因为其中的对象是序列化和反序列化的,但是使用setValueAt()更改表中的值成功了。@ErickQ:不客气,但如果您创建和使用并行数据集合(ArrayList和TableModel),您将面临危险。如果这是我的问题,我会创建方法来序列化和反序列化模型的数据核心,这些方法可以通过getDataVector()
获得,或者使用ArrayList作为AbstractTableModel的核心。谢谢,我必须更改ArrayList中的值,因为其中的对象是序列化和反序列化的,但是使用setValueAt()更改表中的值是有效的。@ErickQ:不客气,但是如果您创建和使用并行数据集合(ArrayList和TableModel),您将面临危险。如果这是我的问题,我会创建方法来序列化和反序列化模型的数据核心,这些数据核心可以通过getDataVector()
获得,或者使用ArrayList作为AbstractTableModel的核心。
JButton btnVender = new JButton("Vender");
btnVender.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
String n = JOptionPane.showInputDialog("Qué producto quieres vender?");
int x=1;
for(Product p : stock){
if(p.getNombre().equals(n)){
x=0;
String numero = JOptionPane.showInputDialog("Cuántos quieres vender?");
int num = Integer.parseInt(numero);
p.decrementar(num);
tableModel.fireTableDataChanged();
table.revalidate();
table.repaint();
break;
}
}
if(x==1){
JOptionPane.showMessageDialog(null, "No existe ese producto");
}
}
});