java - 无法使用 setValueAt 使用自定义 TableModel 更改 JTable 中的值

标签 java swing

我正在编写小的访问时间表。 它基于 JTable,自定义 TableModel 扩展了 AbstractTableModel,如下图所示。

问题是可以初始化 Table 并获得所需的外观 - 但在使用 setValueAt() 更改数据后,Table 的外观没有变化。单击按钮获取示例字符串并在 9:00 设置“类型”和“状态”列中的字段(更精确地取决于给定的预订时间和日期)

此外,我能够在表的末尾插入新行,但无法更新可见的现有值。尝试了很多解决方案,但遗憾的是没有结果。 如果有任何建议,我将不胜感激。

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

最佳答案

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

您可以使用 @Override 注释来测试它,例如...

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

生成编译器错误,其中 as

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

没有。

因此将 setValueAt 方法更改为...

@Override
public void setValueAt(Object value, int row, int col) {
    if (value instanceof String) {
        String[] temp = data.get(row);
        temp[col] = value.toString();

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

为我找到工作

此外,您真的应该避免使用 null 布局,像素完美布局是现代 ui 设计中的一种错觉。影响组件单个尺寸的因素太多,没有一个是您可以控制的。 Swing 旨在与核心的布局管理器一起工作,丢弃这些将导致无穷无尽的问题和问题,您将花费越来越多的时间来尝试纠正

如评论中所述,您还可以使用 public void setValueAt(String value, int row, int col) {(无覆盖注释)并且您的代码可以工作,但我不会调用方法 setValueAt 因为它会产生混淆(正如您所发现的)并排除了您所拥有的只是 TableModel

实例的情况

关于java - 无法使用 setValueAt 使用自定义 TableModel 更改 JTable 中的值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34537697/

相关文章:

java - 从方法中收到错误的颜色

java - 如何将垂直滚动条添加到 Swing 中的 JTextArea?

Java2d : Set gradient for a lines

java - Spring Data Neo4j SDN 4 审计支持

java - 获取一个dom节点的属性

java - session 超时不起作用 vaadin

java - JSlider -- 需要建议

java - 将参数设置为 IN 表达式的列表

java - GWT super 开发模式

java - 将特定于操作系统的窗口渲染到 BufferedImage