我在使用自定义AbstractListModel管理ArrayList时遇到了麻烦。 这是我的模特:
TradesListModel.class
package window;
import java.util.ArrayList;
import javax.swing.AbstractListModel;
import main.Trade;
public class TradesListModel extends AbstractListModel<Object>{
/**
*
*/
private static final long serialVersionUID = 1L;
private ArrayList<Trade> trades;
public TradesListModel(){
trades = new ArrayList<Trade>();
}
public void add(Trade trade){
trades.add(0, trade);
this.fireIntervalAdded(this, 0, 1);
}
public void remove(){
trades.remove(trades.size()-1);
this.fireIntervalRemoved(this, trades.size(), trades.size());
}
public void clear(){
trades.clear();
this.fireIntervalRemoved(this, 0, trades.size());
}
@Override
public int getSize() {
return trades.size();
}
@Override
public Object getElementAt(int index) {
return trades.get(index);
}
}
正如您所看到的,三个并行的线程将每个对象添加到列表的顶部。每个线程通过调用我的主类中的方法来添加新对象:
public void addTrade(Trade trade){
trades.add(trade);
SwingUtilities.invokeLater(new Runnable(){
@Override
public void run() {
myModel.add(trade);
}
});
}
我有另一个newSingleThreadScheduledExecutor
,每隔30秒调用我的主类remove()
的方法,从我的自定义模型处理的列表中删除对象:
public boolean updateTrades(){
Iterator<Trade> it = trades.iterator();
try {
synchronized(this){
index = 0;
while(it.hasNext()){
Trade current = it.next();
if(TimeUnit.MILLISECONDS.toSeconds(System.currentTimeMillis()-current.getTime())>TIME){
it.remove();
SwingUtilities.invokeLater(new Runnable(){
@Override
public void run() {
lm.remove(index);
}
});
}
index++;
}
}
return true;
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return false;
}
trades
列表等于模型中的trades
列表。
我所做的是打印传递给模型的索引值,但它与主类的索引不对应,因为我多次得到此异常:
Exception in thread "AWT-EventQueue-0" java.lang.ArrayIndexOutOfBoundsException: -6
at java.util.ArrayList.elementData(ArrayList.java:418)
at java.util.ArrayList.remove(ArrayList.java:495)
at window.TradesListModel.remove(TradesListModel.java:28)
at main.TradeFinder$2.run(TradeFinder.java:119)
at java.awt.event.InvocationEvent.dispatch(InvocationEvent.java:311)
at java.awt.EventQueue.dispatchEventImpl(EventQueue.java:756)
at java.awt.EventQueue.access$500(EventQueue.java:97)
at java.awt.EventQueue$3.run(EventQueue.java:709)
at java.awt.EventQueue$3.run(EventQueue.java:703)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(ProtectionDomain.java:76)
at java.awt.EventQueue.dispatchEvent(EventQueue.java:726)
at java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:201)
at java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:116)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:105)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:101)
at java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:93)
at java.awt.EventDispatchThread.run(EventDispatchThread.java:82)
TradeFinder
是我的主要课程。
答案 0 :(得分:2)
让我们考虑以下示例:当列表中有三个交易时调用updateTrades
:[t0, t1, t2]
并且应删除所有交易。
在这种情况下,您将调用lm.remove
方法三次:使用索引0
,1
和2
。在第一个列表之后,列表将包含[t1, t2]
,在第二个之后:[t2]
。当您第三次调用它时,没有索引为2
的元素,因此您将获得IndexOutOfBoundsException
。
要解决此问题,只需拨打lm.remove(0)
三次即可。这将按预期删除前三个元素。