我在使用自定义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是我的主班。

最佳答案

让我们考虑下面的示例:当列表中有三笔交易时,调用updateTrades[t0, t1, t2],并且应删除所有这些交易。

在这种情况下,您将调用lm.remove方法三次:索引为012。在第一个列表之后将包含[t1, t2],第二个列表之后将包含[t2]。当您第三次调用它时,没有索引为2的元素,因此您将得到IndexOutOfBoundsException

要解决此问题,只需调用lm.remove(0) 3次。正如预期的那样,这将删除前三个元素。

07-24 09:49
查看更多