ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

使用ReentrantLock与wait()notifyAll对比实现线程通讯-生产者消费者模式

2021-03-05 17:04:56  阅读:175  来源: 互联网

标签:Thread int notifyAll ReentrantLock list 线程 lock new public


实现目标:
1.ReentrantLock
建立一个容器,10个消费者,2个生产者,消费者消费数据,生产者生产数据。
容器中数据达到10,生产者等待,消费者进行消费
容器中数据为0,消费者等待,生产者进行生产

public class TestContainer<W> {
    List<W> list=Collections.synchronizedList(new ArrayList());
    final static int MAX=10;
    static int count=0;
    Lock lock = new ReentrantLock();
    Condition producer = lock.newCondition();
    Condition consumer = lock.newCondition();
    //producer
    public void put(W w){
        try {
            lock.lock();
            while (list.size()==MAX){
                producer.await();
            }
            list.add(w);
            count++;
            consumer.signalAll();
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }
    //consumer
    public W get(){
        W w=null;
        try {
            lock.lock();
            while (list.size()==0){
                consumer.await();
            }
            list.remove(0);
            count--;
            producer.signalAll();
        } catch (Exception e) {
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
        return w;
    }

    public static void main(String[] args) {
        TestContainer<String> c = new TestContainer();
        //10个消费者
        for (int i = 0; i <10 ; i++) {
            new Thread(()->{
                for (int j = 0; j <5 ; j++) {
                    System.out.println(Thread.currentThread().getName()+"---"+j);
                    c.get();
                }
            },"consumer"+i).start();
        }
        try {
            Thread.sleep(1000);
        } catch (Exception e) {
            e.printStackTrace();
        }
        //2个生产者
        for (int i = 0; i <2 ; i++) {
            new Thread(()->{
                for (int j = 0; j <30 ; j++) {
                    System.out.println(Thread.currentThread().getName()+"---"+j);
                    c.put("1111");
                }
            },"producer"+i).start();
        }
    }
}

2.wait()notifyAll实现上述目标
此代码存在问题,生产者wait(),让出锁,有可能生产者再次抢到锁,执行多余代码。
消费者也有此情况。

public class WaitNotifyAll {
    final private LinkedList list = new LinkedList<>();
    Object o=new Object();
    final int MAX=10;
    int count=0;

    public void put(){
        synchronized (this){
            while(list.size()==MAX){
                try {
                    this.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            list.add(o);
            ++count;
            this.notifyAll();
        }
    }
    public void get(){
        synchronized (this){
            while(list.size()==0){
                try {
                    this.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            list.removeFirst();
            count--;
            this.notifyAll();
        }
    }
    public static void main(String[] args) {
        WaitNotifyAll c = new WaitNotifyAll();
        //消费者
        for (int i = 0; i <10 ; i++) {
            new Thread(()->{
                for (int j = 0; j <5 ; j++) {
                    System.out.println(Thread.currentThread().getName()+"----"+j);
                    c.get();
                }
            },"c").start();
        }

        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        //生产者
        for (int i = 0; i <2 ; i++) {
            new Thread(()->{
                for (int j = 0; j <30 ; j++) {
                    System.out.println(Thread.currentThread().getName()+"----"+j);
                    c.put();
                }
            },"p").start();
        }
    }
}

标签:Thread,int,notifyAll,ReentrantLock,list,线程,lock,new,public
来源: https://blog.csdn.net/weixin_39276203/article/details/114402002

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有