0 0

多线程 java.util.ConcurrentModificationException 问题30

目标:定义一个Producer类创建Student并添加到studnts集合中,再定义一个Consumer类从studnts集合中取出数据并打印控制台。Producer,Consumer类都实现了Runnable接口。

在MainTest类的main方法中创建线程并启动,如下所示:

        Producer p = new Producer();
        Thread tp1 = new Thread(p);
        tp1.start();
       
        Consumer c = new Consumer();
        Thread tc1 = new Thread(c);
        tc1.start();

这个时候,当tc1线程要执行的时候就报java.util.ConcurrentModificationException错误,我本是对线程调用这一块儿不是太明白。请大侠们帮我解决一下儿,请告诉我问啥会出这样的问题。

 

具体类如下所示:

 

 

Student类:一个普通类,其中定义了一个name属性和一个age属性。

public class Student {
    private String name;
    private String age;
   
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getAge() {
        return age;
    }
    public void setAge(String age) {
        this.age = age;
    }
   
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + ((age == null) ? 0 : age.hashCode());
        result = prime * result + ((name == null) ? 0 : name.hashCode());
        return result;
    }
    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;
        if (obj == null)
            return false;
        if (getClass() != obj.getClass())
            return false;
        Student other = (Student) obj;
        if (age == null) {
            if (other.age != null)
                return false;
        } else if (!age.equals(other.age))
            return false;
        if (name == null) {
            if (other.name != null)
                return false;
        } else if (!name.equals(other.name))
            return false;
        return true;
    }

   
   
}

 

Classroom接口:其中定义了一个名称为studnts的List用于存放Student对象。

public interface Classroom {
    List<Student> studnts = new ArrayList<Student>();
}

 

State抽象类:其中定义了一个状态,想用这个对studnts集合的操作进行控制。

public abstract class State {
    boolean state=true;
}

 

Producer类:添加Student到studnts中。

public class Producer extends State implements Runnable, Classroom {
    private int i=0;
    @Override
    public void run() {
        while(true) {
           
            if(i<10) {
                saveStudent();
            }
        }       
    }
   
    public synchronized void saveStudent() {
       
        if(!state) {
            try {
                wait();
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
       
        Student s = new Student();
        s.setName("name"+i);
        s.setAge("age"+i++);
        studnts.add(s);
       
        System.out.println(Thread.currentThread().getName()+" --- Add Student "+s.getName()+" -- "+s.getAge());
       
        state=true;
        notify();
    }

}

Consumer类:从studnts中取出strut对象并打印出来。

public class Consumer extends State implements Runnable, Classroom {
   
    @Override
    public void run() {
        while(true) {
            getStudent();
        }
    }
   
    public synchronized void getStudent() {

        if(!state) {
            try {
                wait();
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }   
           
        for(Student s : studnts) {
            System.out.println(Thread.currentThread().getName()+" --- get Stduent "+s.getName()+" --- "+s.getAge());
        }
       
        state = false;
       
        notify();
   
    }

}

 

 

1个答案 按时间排序 按投票排序

0 0

采纳的答案

Classroom里定义的students是静态的,因此会被Producer和Consumer共享。Producer和Consumer在各自的线程里同时访问students,但是又没有同步(这两个类里是在方法上加synchronized,即在类实例上做同步,Producer和Consumer是不同实例,因此他们是不同步的), 所以会造成ConcurrentModificationException。

解决的方法是:不在方法上加synchronized,而是使用synchronized (students)同步块来同步Producer和Consumer里的操作。

另外最好不要将students当作静态变量,这样让人不好理解,可以将students作为构造函数的参数传入producer和Consumer。

2013年2月14日 14:20

相关推荐

    出现java.util.ConcurrentModificationException 问题及解决办法

    在Java编程中,`java.util.ConcurrentModificationException` 是一个常见的运行时异常,通常发生在尝试并发修改集合时。这个异常的产生是由于集合类(如HashMap)的非线程安全特性,当你在一个线程中使用迭代器遍历...

    java.util.ConcurrentModificationException 解决方法

    `java.util.ConcurrentModificationException` 是一个在 Java 中常见的运行时异常,它通常发生在多线程环境中,当一个线程正在遍历一个集合(如 `ArrayList`, `HashMap` 等),而另一个线程同时尝试修改这个集合时。...

    java 集合并发操作出现的异常ConcurrentModificationException

    在Java编程中,`ConcurrentModificationException`是一个常见的运行时异常,主要出现在多线程环境下对集合类(如List、Set、Map等)进行并发修改时。然而,这个异常不仅限于多线程环境,即使在单线程中,如果在遍历...

    Java语言的Util类详细介绍

    但是,因为Vector是同步的,当一个Iterator被创建而且正在被使用,另一个线程改变了Vector的状态(例如,添加或删除元素),Iterator将抛出ConcurrentModificationException异常。 Set接口是Collection接口的子接口...

    Java多线程安全集合

    理解并熟练运用这些线程安全集合是构建健壮、高性能的多线程Java应用程序的基础。它们能帮助开发者编写出更安全、更高效的代码,避免因并发问题导致的错误。同时,根据具体场景选择合适的集合类型,可以极大地提高...

    多线程中使用Java集合类.doc

    在多线程环境中,使用Java集合类需要注意线程安全问题,因为不同的线程可能会同时访问和修改集合,导致数据不一致或引发异常。本篇主要讨论在并发编程中使用Java集合类的一些关键知识点。 首先,Java集合框架中的大...

    JAVA.BUG模式详解

    在多线程环境中,不正确的同步可能导致数据不一致、死锁或活锁等问题。使用`synchronized`关键字、`volatile`变量、`java.util.concurrent`包中的工具类等可以有效地管理并发。 四、内存泄漏 Java中的内存泄漏并不...

    多线程程序避免冲突的3条简单规则

    4. `java.util.concurrent.atomic`包:提供了一组原子类,如`AtomicInteger`、`AtomicLong`等,它们的原子操作在多线程环境中避免了锁的开销。 遵循以上规则,可以显著提高多线程程序的并发性能和稳定性。然而,...

    jdk 1.6 API 中文版帮助文档

    - `java.util.concurrent`: 引入并发集合,如`ConcurrentHashMap`、`CopyOnWriteArrayList`,支持多线程环境下的高效操作。 - `java.util.Set`和`java.util.Map`接口的实现:添加了`LinkedHashSet`和`LinkedHashMap`...

    Java实验五线程.pdf

    同时,`java.util.concurrent`包提供了更高级的线程管理工具,如`ExecutorService`、`Future`、`Semaphore`等,用于更复杂和高效的并发编程。 总之,这个实验旨在帮助学习者理解Java中线程的概念,以及如何创建、...

    Java多线程基础-02、数组定义方式一、访问、注意事项.rar

    在Java编程语言中,多线程是程序设计中的一个重要概念,尤其在处理高并发和资源优化的场景下显得尤为重要。本教程将深入讲解Java多线程的基础知识,重点关注数组的定义方式、访问方法以及在多线程环境下的注意事项。...

    concurrent.rar

    Java并发包,也被称为`java.concurrent`包,是Java编程语言中处理多线程和并发操作的核心工具包。它提供了一系列高效、线程安全的类和接口,使得开发者能够更安全、更容易地编写多线程应用程序。这个压缩包`...

    遍历并批量删除容器中元素出现ConcurrentModificationException原因及处置

    在Java编程中,`ConcurrentModificationException`是一个常见的运行时异常,通常发生在多线程环境下对集合进行迭代和修改操作时。此问题的核心在于,Java的集合类(如ArrayList、LinkedList、HashSet等)并不支持...

    java_util_concurrent_user_guide并发工具包详解

    Java并发工具包(java.util.concurrent)是Java编程中不可或缺的一部分,它为多线程环境提供了高效、安全且易用的工具。这个包包含了各种类和接口,帮助开发者编写高效的并发程序,避免了直接操作线程所带来的复杂性...

    Java 7编程高级进阶

    当线程处于等待状态时,Java 7可以生成不包含阻塞信息的堆栈跟踪,这有助于提高性能并减少诊断问题时的资源消耗。 8. **改进的集合接口(Multimaps 和 Multisets)** 虽非Java 7标准库的一部分,但Google的Guava...

    Java 实例 - 只读集合源代码+详细指导教程.zip

    在多线程环境中,只读集合特别有用,因为它们能防止并发修改异常(`ConcurrentModificationException`)。当多个线程试图同时修改集合时,可能会出现这种问题。只读集合确保了即使在并发环境下,数据也能保持一致。 ...

    高级程序员必会的HashMap的线程安全问题,适用于0~2年的.7z

    然而,对于多线程环境,HashMap并不是线程安全的,这在并发编程中可能会引发一系列问题。本篇将深入探讨HashMap的线程安全问题,并提供相关的解决方案。 首先,我们需要了解HashMap在多线程环境下可能出现的问题: ...

Global site tag (gtag.js) - Google Analytics