时间计数器以秒为单位显示表中的年龄。理想情况下,它将每秒更新一次。我知道我可以在表模型中增加适当的数据,触发事件(每行一个),等等。这似乎有点过头了!有没有更好,更轻松的方法?
最佳答案
您需要的是:
一种在表格模型中建模行的“年龄”的方法。最好以隐式方式进行此操作,因此您只需将行的创建时间存储一次,并在请求单元格值时计算年龄(对象getValueAt(row,column))。
一个(守护程序)线程,它每秒触发一次表更改事件。您不必每行触发一个事件,而是可以触发一个信号,该事件表示整个列的更新。
这是表模型的一些伪代码:
public Object getValueAt (int rowIndex, int columnIndex) {
// if it's the column with the 'row age', calculate the age and return it
long rowAgeMs = System.currentTimeMs() - getCreationTime(rowIndex);
// return the age in milliseconds, or a date, or a formatted time string
}
然后,表模型还应该为线程提供一种方法,以便它可以为“行龄”列触发更改事件:
公共类MyTableModel实现TableModel {
private final List<TableModelListener> listeners = new LinkedList<TableModelListener>();
public void addTableModelListener (TableModelListener l) {
listeners.add(l);
}
public void removeTableModelListener (TableModelListener l) {
listeners.remove(l);
}
public void updateColumn (int column) {
TableModelEvent evt = new TableModelEvent(this, 0, Math.max(0, getRowCount() - 1), column);
for (TableModelListener listener : listeners) {
listener.tableChanged(evt);
}
}
然后,该线程将仅每秒为“行年龄”列触发updateColumn(..)方法。此方法的调用应在EventDispatchThread中完成,可以使用SwingUtilities.invokeAndWait(..)或SwingUtilities.invokeLater(..)完成。
Thread rowAgeUpdater = new Thread() {
@Override
public void run () {
while (isAlive()) {
try {
long time = System.currentTimeMillis();
long sleepTime = (time / 1000 + 1) * 1000 - time;
Thread.sleep(sleepTime);
SwingUtilities.invokeAndWait(new Runnable() {
public void run () {
model.updateColumn(ROW_AGE_COLUMN_INDEX);
}
});
} catch (Exception e) {
return;
}
}
}
};
rowAgeUpdater.setDaemon(true);
rowAgeUpdater.setPriority(Thread.MIN_PRIORITY);
rowAgeUpdater.start();
只要TableModelEvent的粒度仅覆盖需要更新的单元格(在您的情况下:仅包含行龄的列),这是实现此目的的最有效方法。