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.util.ConcurrentModificationException(解决方案).md

    Java的`java.util.concurrent`包中提供了多种线程安全的集合类,如`ConcurrentHashMap`、`CopyOnWriteArrayList`等,它们能够在多线程环境下安全地修改集合内容。 - **CopyOnWriteArrayList**:这种实现方式通过...

    java.util.concurrent系列文章(1)

    `java.util.concurrent` 包中的 `ConcurrentHashMap` 和 `CopyOnWriteArrayList` 为 Java 开发者提供了强大的工具来处理并发编程中的常见问题。这些类的设计考虑了高性能和线程安全性之间的平衡,使得开发者能够在多...

    java.util介绍.pdf

    - **同步机制**:由于`Vector`内部实现了同步机制,因此在多线程环境中更加安全。然而,这也意味着`Vector`在单线程环境下的性能可能会低于`ArrayList`。 - **注意事项**:使用`Vector`创建的迭代器时,如果其他...

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

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

    com.harmonyos4.exception.ConcurrentModificationException.md

    `ConcurrentModificationException`是一个多线程环境下常见的异常。在进行迭代集合的操作时,如果尝试修改正在迭代的集合结构,如添加或删除元素,Java虚拟机(JVM)会抛出此异常。这个异常对于鸿蒙开发同样适用。要...

    Java语言的Util类详细介绍

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

    Java多线程安全集合

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

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

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

    ConcurrentModificationException如何解决.md

    在多线程编程中,对集合进行迭代时,如果同时存在对集合结构的修改,比如添加或删除元素,那么就可能会引发`ConcurrentModificationException`异常。这种异常属于运行时异常,通常出现在迭代器与集合本身的状态不...

    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.md

    1. ConcurrentModificationException是一个在Java中多线程环境下修改集合时可能出现的异常,它通常发生在使用迭代器遍历集合的过程中,如果有其他线程或代码块修改了集合的内容,则可能会抛出该异常。 2. 解决该...

    ConcurrentModificationException(解决方案).md

    在多线程编程中,我们经常会遇到一个运行时异常 ConcurrentModificationException。该异常通常是由于一个线程在对集合进行迭代的过程中,另一个线程修改了这个集合的结构,例如添加或删除元素。这种情况下,会出现...

Global site tag (gtag.js) - Google Analytics