[英]java JTable replaced (?) but not being refreshed despite revalidate and repaint
Having SSCCE as below, and assuming You are on GNU/Linux, please see if You can answer my question below. 拥有如下所示的SSCCE,并假设您使用的是GNU / Linux,请查看您是否可以在下面回答我的问题。 Save two files to "/tmp/aaa/some/packagepackagea/Test.java" and "/tmp/aaa/some/packagepackagea/DataHandler.java" or fix paths.
将两个文件保存到“ /tmp/aaa/some/packagepackagea/Test.java”和“ /tmp/aaa/some/packagepackagea/DataHandler.java”或修复路径。 I've tested that it compiles and runs.
我已经测试过它可以编译并运行。
Please fill table with your data, or press "save" (hardcoded paths ("/tmp/aaa/testdata")) 请用您的数据填写表格,或按“保存”(硬编码路径(“ / tmp / aaa / testdata”))
final JTable table
? final JTable table
吗? If I understand it correctly, if table is final then d.load("/tmp/aaa/test", table, ";");
d.load("/tmp/aaa/test", table, ";");
passes a copy (or a reference?) to DataHandler (d) load function. package some.packagepackagea;
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.text.*;
import java.util.*;
import java.text.*;
public class Test {
Test () {
final DataHandler d = new DataHandler();
JFrame root = new JFrame("zxczxc");
root.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
final javax.swing.table.DefaultTableModel model = new javax.swing.table.DefaultTableModel();
final JPanel x = new JPanel();
final JTable table = new JTable(model);
JScrollPane table_container = new JScrollPane(table);
// read some columns from file/db
final String cols[] = {"aaaa", "bbbbb", "ccccc", "ddddd", "eeee"};
for (String col:cols)
model.addColumn(col);
Object y[] = new Object[]{"a", "b"};
model.addRow(y);
model.addRow(y);
JButton a = new JButton("save");
JButton b = new JButton("load");
x.add(a);
x.add(b);
x.add(table_container);
a.addActionListener(new java.awt.event.ActionListener() {
public void actionPerformed(ActionEvent e) {
d.save(table, "/tmp/aaa/testdata", ";");
x.revalidate();
x.repaint();
}
});
b.addActionListener(new java.awt.event.ActionListener() {
public void actionPerformed(ActionEvent e) {
d.load("/tmp/aaa/testdata", table, ";");
model.fireTableDataChanged();
x.revalidate();
x.repaint();
}
});
root.add(x);
root.pack(); // shrink borders
root.setLocationRelativeTo(null); // center panel
root.setVisible(true);
}
public static void main(String[] args) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() {
Test it = new Test();
}
});
}
}
package some.packagepackagea;
import java.io.*;
import java.nio.CharBuffer;
import javax.swing.*;
import javax.swing.table.*;
import java.util.ArrayList;
public class DataHandler {
public void save(JTable table, String file, String separator){
File f = new File(file);
save(table, f, separator);
}
public void save(JTable table, File file, String separator){
try {
TableModel tableModel = table.getModel();
FileWriter disk = new FileWriter(file);
for(int i = 0; i < tableModel.getColumnCount(); i++){
disk.write(tableModel.getColumnName(i) + separator);
}
disk.write("\n");
for(int i=0; i< tableModel.getRowCount(); i++) {
for(int j=0; j < tableModel.getColumnCount(); j++) {
Object item = tableModel.getValueAt(i,j);
if (null != item)
disk.write(item.toString() + separator);
}
disk.write("\n");
}
disk.close();
((DefaultTableModel)tableModel).getDataVector().removeAllElements();
} catch (IOException e){ System.out.println(e); }
}
public void load(String file, JTable table, String separator){
File f = new File(file);
load(f, table, separator);
}
public void load(File file, JTable table, String separator){
CharBuffer buf = CharBuffer.allocate((int)file.length());
try {
FileReader disk = new FileReader(file);
disk.read(buf);
buf.rewind();
disk.close();
} catch (IOException e){ System.out.println(e); }
String rows[] = (buf.toString()).split("\n");
ArrayList<String[]> cols = new ArrayList<String[]>();
for(String row:rows){
cols.add(row.split(separator));
}
// it might be more efficient to return model and initialize table with
// it but i find it nicer when methods are somewhat symmetric.
TableModel tableModel = new DefaultTableModel();
for(String[] col:cols){
((DefaultTableModel)tableModel).addRow(col);
}
table = new JTable(tableModel);
((DefaultTableModel)tableModel).fireTableDataChanged();
}
}
EDIT: 编辑:
I've tried also to replace 我也试过更换
table = new JTable(tableModel);
with 与
table.setModel(tableModel);
but that didn't do the trick as well. 但这并不能解决问题。
You are creating a new table which isn't replacing referenced by the GUI. 您正在创建一个不会替换GUI引用的新表。
So rewriting the load method a bit you could just set the model on the table, somthing like this: 因此,稍微重写一下load方法,您可以在表上设置模型,就像这样:
String rows[] = (buf.toString()).split("\n");
DefaultTableModel tableModel = new DefaultTableModel();
for (String col:rows[0].split(separator))
tableModel.addColumn(col);
for (int i = 1; i < rows.length; i++)
tableModel.addRow(rows[i].split(separator));
table.setModel(tableModel);
The second thing is your save method. 第二件事是您的保存方法。 You are removing
null
values when outputting it to disk. 将其输出到磁盘时,您正在删除
null
值。 Instead of: 代替:
Object item = tableModel.getValueAt(i,j);
if (null != item)
disk.write(item.toString() + separator);
Do something like this: 做这样的事情:
Object item = tableModel.getValueAt(i,j);
disk.write((item != null ? item.toString() : "") + separator);
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.