java - 获取与线程和数组列表相关的错误

标签 java multithreading arraylist

您好,我正在编写一个非常简单的游戏。玩家可以使用鼠标移动宇宙飞船,每 200 毫秒发射一次新光束。这个光束在 while(true) 循环中移动,当它的 y 为 0 或 400(帧的边界)时,我使用 break 来结束循环(和线程)。每个光束都有自己的线程。还有在背景中移动的星星。他们每个人都像梁一样移动并拥有自己的线程。因此,正如您所看到的,经常从 arrayLists 中添加和删除。一切正常,但有时我会收到这样的错误:

Exception in thread "AWT-EventQueue-0" java.util.ConcurrentModificationException
    at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:819)
    at java.util.ArrayList$Itr.next(ArrayList.java:791)
    at spacecommander.MainPanel.paintComponent(MainPanel.java:50)

它们在游戏中不会造成任何问题,但我该如何消除它们?也许我应该使用同步之类的东西?

编辑:这是代码

public class MainPanel extends JPanel {
    private Player player = new Player(100, 100, 3, 3);
    private Point2D targetPoint = new Point2D.Float(130, 350); //Poczatkowa pozycja statku
    private ArrayList<Beam> beams = new ArrayList<Beam>();
    private InputMap imap = getInputMap(JComponent.WHEN_IN_FOCUSED_WINDOW);
    private ActionMap amap = getActionMap();
    private Random rand = new Random();

    public MainPanel() {
        setPreferredSize(new Dimension(300, 400));

        addMouseMotionListener(new MouseMotionHandler());

        //Rozpoczynanie watkow
        Thread t = new Thread(new PlayerMoveRunnable());
        t.start();
        Thread t2 = new Thread(new PlayerShootRunnable());
        t2.start();
    }

    public void paintComponent(Graphics g) {
        Graphics2D g2 = (Graphics2D)g;
        g2.setColor(Color.BLACK);
        g2.fillRect(0, 0, 300, 400);
        //Rysowanie gracza
        g2.drawImage(player.getImage(), (int)player.getX(), (int)player.getY(), null);
        //Rysowanie pociskow
        for (Beam beam : beams) {
            g2.drawImage(beam.getImage(), (int)beam.getX(), (int)beam.getY(), null);
        }
    }

    public void makeShortcut(String name, String keys, AbstractAction action) {
        imap.put(KeyStroke.getKeyStroke(keys), name);
        amap.put(name, action);
    }

    //Watek dziala caly czas bo gracz i tak caly czas sie rusza
    private class PlayerMoveRunnable implements Runnable {
        public void run() {
            try {
                while (true) {
                    player.moveToPoint(targetPoint);
                    repaint();
                    Thread.sleep(15);
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    //Takze dziala caly czas. Dodaje nowy pocisk co 200ms
    private class PlayerShootRunnable implements Runnable {
        public void run() {
            try {
                while (true) {
                    //Wybranie pocisku do wystrzelenia w zaleznosci od mode gracza
                    Thread t;
                    switch (player.getBeamMode()) {
                    case 1:
                        t = new Thread(new BeamMoveRunnable(new Beam1(100, 100, 10, 10, 10)));
                        break;
                    }
                    t.start();
                    Thread.sleep(200);
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    private class BeamMoveRunnable implements Runnable {
        private Beam beam;

        public BeamMoveRunnable(Beam beam) {
            this.beam = beam;
        }

        public void run() {
            Beam beam = this.beam;
            beams.add(beam);
            try {
                while (true) {
                    if (beam.getY() <= 0) {
                        beams.remove(beam);
                        break;
                    }
                    beam.move();
                    repaint();
                    Thread.sleep(20);
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }

    private class MouseMotionHandler extends MouseAdapter {
        public void mouseMoved(MouseEvent event) {
            targetPoint = event.getPoint();
        }
    }
}

最佳答案

正如您所怀疑的,这似乎是一个同步问题。可能当您的绘图代码正在迭代 beams 列表时,BeamMoveRunnable 同时修改列表(添加或删除光束),并导致 ConcurrentModificationException。就个人而言,我不会使用单独的线程来移动光束,而是使用一个简单的循环,它首先更新游戏(一次移动所有光束等),然后重新绘制屏幕。但是,您也可以只同步对列表的访问,这样一次只有一个线程可以访问它。

关于java - 获取与线程和数组列表相关的错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14171331/

相关文章:

java - Java 中的线程和观察者模式

c++ - 在类似于 unix shell 脚本的 c++ 中调用后台函数

java - Arraylist.sort 请求比较器

java - 使用值对 ArrayList<HashMap<String, String>> 进行排序

java - 检查对象是否已存在于 Java ArrayList 中

Java Map迭代使用Map Inside Map并在迭代后将其存储在另一个Map中

java - Spring Data JPA - 删除子实体而不是在更新时设置为空?

c++ - 多线程和临界区使用 - C++

java - 在 web.xml 中以编程方式禁用过滤器

java - 在浏览器中直接点击 post 方法 url 时显示错误消息