同步对象从反向列表中删除[JList - SWING]

时间:2016-11-04 23:40:14

标签: java multithreading swing jlist

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

1 个答案:

答案 0 :(得分:2)

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

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

要解决此问题,只需拨打lm.remove(0)三次即可。这将按预期删除前三个元素。