1

我的 JTable 有问题。我的 JTable 显示数据库的内容。一个数据库表具有名称类别。每个类别都显示在 JComboBox 中。如果我单击一个类别,它应该更新表格内容。

这是我为您准备的代码的简短片段,因此更容易帮助我。代码应该是可运行的:

(TestClass - 主要)

package test;

import java.awt.BorderLayout;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.LinkedList;

import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;

public class TestClass implements ActionListener{

    String[] header = {"head", "head", "head"};
    Object[][] data = {{Boolean.FALSE, "text", "text"}, {Boolean.FALSE, "text", "text"}, {Boolean.FALSE, "text", "text"}};

    LinkedList<String> newdata = new LinkedList<String>();

    String[] combolist = {"apple", "banana", "cranberry"};

    JComboBox<String> combobox = new JComboBox<String>(combolist);
    JTable table = new JTable(new TestTableModel(data, header));
    JFrame frame = new JFrame();
    JPanel panel = new JPanel(new GridLayout(1, 0, 1, 0));

    public TestClass() {
        combobox.addActionListener(this);
        panel.add(combobox);

        frame.add(panel, BorderLayout.NORTH);
        frame.add(new JScrollPane(table), BorderLayout.CENTER);

        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        frame.setLocationByPlatform(true);
        frame.setVisible(true);
        frame.pack();
    }

    @Override
    public void actionPerformed(ActionEvent e) {
        if (e.getSource() == this.combobox) {
            JComboBox<String> combobox = this.combobox;

            newdata.add("Test1");
            newdata.add("Test2");

            TestTableModel model = (TestTableModel) table.getModel();

            int i = 0;
            for (String text : newdata) {
                data[i][0] = Boolean.TRUE;
                data[i][1] = text;
                data[i][2] = text;
                i++;
            }

            model.setData(data);
        }
    }

    public static void main(String[] args) {
        new TestClass();
    }
}

(TestTableModel - AbstractTableModel 如果需要(你需要它来执行代码!))

package test;

import javax.swing.table.AbstractTableModel;

public class TestTableModel extends AbstractTableModel {
    private static final long serialVersionUID = 5044877015250409328L;

    private String[] header;
    private Object[][] data;

    public TestTableModel(Object[][] data, String[] header) {
        this.header = header;
        this.data = data;
    }

    public void setData(Object[][] data) {
        this.data = data;
        fireTableDataChanged();
    }

    @Override
    public Class<?> getColumnClass(int column) {
        if (column == 0) {
            return Boolean.class;
        }
        return super.getColumnClass(column);
    }

    @Override
    public int getColumnCount() {
        return header.length;
    }

    @Override
    public String getColumnName(int column) {
        return header[column];
    }

    @Override
    public int getRowCount() {
        return data.length;
    }

    @Override
    public Object getValueAt(int row, int column) {
        return data[row][column];
    }

    @Override
    public boolean isCellEditable(int row, int column) {
        return column == 0;
    }

    @Override
    public void setValueAt(Object value, int row, int column) {
        data[row][column] = value;
    }

}

使用此短代码,如果您更改类别,表格将冻结。在我的整个代码中,它也冻结了,但是当我调整窗口大小时,我能够在后台看到更新的表格(表格大小调整为与框架相同的大小)。我不知道为什么它不在剪断处。

编辑:更改内容的问题已解决。源已更新。但是获得正确的表大小的问题还没有解决。在源代码中,我首先使用 3 行数组,然后使用 2 行数组。我想删除旧表并创建一个新表,所以行大小是正确的。

基本上我只想用新内容更新表格。

  • 谢谢你的帮助!
4

2 回答 2

1

代码有问题,因为每次actionPerformed()调用时,您都在创建一个新组件:

table = new JTable(new TestTableModel(data, header));
frame.add(new JScrollPane( table ));  // <-- BTW: here you need to put the table otherwise you are adding an empty JScrollPane
frame.validate();

(注意:还有一个额外的错误,@mKorbel 提到了它)。

但是,您已经在框架中添加了JScrollPaneaJTable并且它们继续存在。(如果您尝试调整窗口大小,您将在旧表格下方看到新表格)。

更新表格数据的正确方法是对其进行模型中TableModel所需的任何修改,然后根据您所做的更改,您将触发适当的方法来通知表格重绘自身。fireXXX()

作为一个粗略的例子,您的代码将是:

@Override
public void actionPerformed(ActionEvent e) {
    if (e.getSource() == this.combobox) {
        JComboBox<String> combobox = this.combobox;

        newdata.clear();    // Clear your list or create a new one otherwise data will keep piling up.  
        newdata.add("Test1");
        newdata.add("Test2");

        TestTableModel model = (TestTableModel) table.getModel();

        // Since you need to replace the whole data create a new Object[][] each time
        Object[][] newModelData = new Object[newdata.size()][3];

        // Set whatever data to the new array
        int i = 0;
        for (String text : newdata) {
            newModelData[i][0] = Boolean.TRUE;
            newModelData[i][1] = text;
            newModelData[i][2] = text;
            i++;
        }

        // replace all data of the current model
        model.setData(newModelData);
    }
}
....

// Now inside your table model:
    ...
    @Override
    public Class<?> getColumnClass(int column) {
    // if (column == 0) {   
    //     return Boolean.class;             // <-- this produces a ClassCastException with the data you aretrying to set
    // }
       return super.getColumnClass(column);
    }

    public void setData(Object[][] data) {
        this.data = data;       //  <-- Update the data
        fireTableDataChanged(); //  <-- fire the event so the table is notified. If you change only one cell you need to call the appropriate fire event
    }
    ...

更新 1:您的新代码问题已经解决了您更新模型中数据的方式。但是,当您更新data结构时,您有一个逻辑缺陷。此变量以 3 行的数组开始。在该方法中,您对只有 2 个条目actionPerformed()的列表长度执行循环。newdata因此,您只更新模型的前 2 行。

更新 2:您似乎没有抓住重点。你如何更新你的模型在这里很重要。该表将显示您的模型拥有的任何数据。如果您只更新 2 行但不修改第 3 行,则表格将显示 3 行(2 行新行和 1 行旧行)。由于您每次都需要更改所有数据,因此您需要完全替换模型中的数据。每次都需要重新创建的是您的数据,而不是表。请参阅更新代码示例。我添加了actionPerformed()使用当前源代码重新初始化数据的方法。请阅读内联评论。

于 2013-08-09T12:21:23.667 回答
-1

更改 jtable 中的数据后尝试以下代码。

模型.fireTableDataChanged(); table.repaint();

其中 model 是 tablemodel 对象,table 是 JTable 对象

我希望它有帮助!!!

于 2013-08-09T11:56:22.940 回答