使用setValueAt更改数据后无法更新JTable的值

时间:2015-12-30 02:28:09

标签: java swing

我已经通过访问完成了JTable的简单计划。 它包含自定义AbstractTableModel,它显示如下所示的三列。

问题在于可以初始化Table并获得所需的外观 - 但在数据更改后,Table的外观没有变化。每个按钮单击从数据库中获取数据,并在列TYPE和STATE中设置字段 - 具体取决于给定的预订时间和日期。

更重要的是,我能够在表的末尾插入新行,但无法使可见的现有值更新。 我已经阅读了几个类似的主题但在我的案例中没有任何帮助 提前感谢您的每一个建议。

enter image description here

[已更新]说明问题的工作代码:

import java.awt.EventQueue;
import java.util.Vector;

import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.table.AbstractTableModel;

import javax.swing.JButton;
import java.awt.event.ActionListener;
import java.awt.event.ActionEvent;

public class ModelTest {

    private JFrame frame;
    private JTable tablePendingVisits;
    private PendingVisitModel pendingVisitModel;
    private JScrollPane scrollPanePendingVisits;

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            public void run() {
                try {
                    ModelTest window = new ModelTest();
                    window.frame.setVisible(true);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }

    public ModelTest() {
        frame = new JFrame();
        frame.setBounds(100, 100, 450, 407);
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.getContentPane().setLayout(null);

        JButton btnChangeValue = new JButton("Change value at 9:00");
        btnChangeValue.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent arg0) {
                refreshTableModel();
            }
        });
        btnChangeValue.setBounds(63, 308, 305, 23);
        frame.getContentPane().add(btnChangeValue);

        tablePendingVisits = new JTable();
        scrollPanePendingVisits = new JScrollPane();
        pendingVisitModel = new PendingVisitModel();

        tablePendingVisits.setModel(pendingVisitModel);
        scrollPanePendingVisits.setBounds(63, 36, 305, 246);
        scrollPanePendingVisits.setViewportView(tablePendingVisits);

        frame.getContentPane().add(scrollPanePendingVisits);
    }

    public void refreshTableModel(){
        String[] sampleString = {"9:00", "Bobby", "Tables"};

        // search for row with 9:00 and replace values in given columns
        for (int i = 0; i < pendingVisitModel.getRowCount(); i++) {

            if( sampleString[0].equals(pendingVisitModel.getValueAt(i, 0)) ) {   // Change row values when both hours are equal 
                pendingVisitModel.setValueAt(sampleString[1], i, 1);             // Change at type column
                pendingVisitModel.setValueAt(sampleString[2], i, 2);             // Change at status column      
            }
        }
    }
}

// Custom TableModel
class PendingVisitModel extends AbstractTableModel {

    private static final long serialVersionUID = 1L;

    private String[] columnNames = {"HOUR", "TYPE", "STATE"};
    private Vector<String[]> data = new Vector<String[]>();

    public PendingVisitModel() {

        for(int i = 8; i<15; i++) {
            data.add(new String[]{i+":00", "-", "Free"} );
            data.add(new String[]{i+":15", "-", "Free"} );
            data.add(new String[]{i+":30", "-", "Free"} );
            data.add(new String[]{i+":45", "-", "Free"} );
        }
    }

    public int getColumnCount() {
        return columnNames.length;
    }

    public int getRowCount() {
        return data.size();
    }

    public String getColumnName(int col) {
        return columnNames[col];
    }

    public String getValueAt(int row, int col) {
        String[] temp = data.get(row);
        if(temp.length > 0 && col < 3)
            return temp[col];
        else
            return null;
    }

    public void setValueAt(String[] value, int row, int col) {
        String[] temp = data.get(row);
        temp[col] = value[col];

        data.set(row, temp);
        fireTableRowsUpdated(row, row);
    }

    public void insertRow(String[] value) {
        data.add(value);
        fireTableRowsInserted(data.size(), data.size());
    }

    public void clearRows(){
        data.clear();
        fireTableDataChanged();
    }

    public void removeAllEntry(){
        data.clear();
        fireTableDataChanged();
    }

    public boolean isCellEditable(int row, int col) {
        return false;
    }

    @Override
    public Class<String> getColumnClass(int colNum) {
        return String.class;
    }
}

1 个答案:

答案 0 :(得分:3)

PendingVisitModel pendingVisitModel = new PendingVisitModel();

在我看来,pendingVisitModel被定义为局部变量。这是您添加到表格中的模型。

refreshTableModel()方法引用了一个pendingVisitModel变量,但我猜这是一个表格未使用的实例变量。

删除pendingVisitModel

的本地实例
fireTableDataChanged();

也不要在所有TableModel方法中继续使用firTableDataChanged。 API提供了更适合不同事件的其他方法。

编辑:

public void setValueAt(String[] value, int row, int col) {

您没有覆盖setValueAt(...)方法。 value参数是Object而不是String数组。

每当覆盖类的方法时,都应该在方法之前使用@Override注释。这样,如果您错误地覆盖方法,编译器会给您一个错误(节省数小时的挫折......)。

@Override
public void setValueAt(Object value, int row, int col) {