java中的生产者消费者模式详解

方式 一: Synchronized方式
注:此种方式会造成资源的浪费:
利用锁的notifyAll()方法会将所有的线程都唤醒,会造成资源的浪费

class Resource {
    private String name;
    private int count = 1;
    private boolean flag = false;

    public synchronized void set(String name) {

        /*
            对于多个生产者和消费者,为什么要用while来判断标记?
            原因: 让要被唤醒的线程再判断一次标记,
                    因为现在的标记可能已经不是它原来等待时的状态了
         */
        while (flag) { 
            try {this.wait();} catch (Exception e) {}
        }
        this.name = name + "--" + count++;
        System.out.println(Thread.currentThread().getName() + "...Producer..." + this.name);
        this.flag = true;
        /*
            为什么要用notifyAll()
            因为需要唤醒对方的线程,如果只用notify()的方式有可能出现只唤醒本方线程的情况
            导致程序中的所有线程都处于等待的状态下!!!
         */
        this.notifyAll();
    }

    public synchronized void out() {
        while (!flag) {
            try {this.wait();} catch (Exception e) {}
        }
        System.out.println(Thread.currentThread().getName() + ".....Consumer....." + this.name);
        this.flag = false;
        this.notifyAll();
    }
}

class Producer implements Runnable {
    private Resource r;
    public Producer(Resource r) {
        this.r = r;
    }
    public void run() {
        while (true) {
            r.set("+Items+");
        }
    }
}

class Consumer implements Runnable {
    private Resource r;

    public Consumer(Resource r) {
        this.r = r;
    }

    public void run() {
        while (true) {
            r.out();
        }
    }
}


public class ProducerAndConsumer {
    public static void main(String[] args) {
        Resource r = new Resource();
        new Thread(new Producer(r)).start();
        new Thread(new Producer(r)).start();
        new Thread(new Consumer(r)).start();
        new Thread(new Consumer(r)).start();
    }
}

方式二:
利用Lock的方式显示的声明锁
这种方式可以人唤醒想要唤醒的线程,能够减少资源的浪费

/*
    JDK1.5中提供了多线程中锁的升级解决方案:
    将同步中的Synchronized替换成显式的lock操作
    一个锁可以对应多个Condition对象
    将object中的wait()、notify()、notifyAll()等方法替换成了Condition对象拥有的方法
    Condition对象通过lock锁的newCondition()方法来获取。
    不同类型的线程用不同的Condition对象来实现线程状态的切换,以达到对不同类型的线程进行不同的操作的目的
    例如:下面的示例中用condition对象来对生产者的线程进行操作,用conditionPro对象来对消费者的线程进行操作,
        这样就可以实现生产者的线程只唤醒消费者,消费者的线程只唤醒生产者,减少资源的浪费。
        即两种类型的线程用的同一把锁,但是用的不同的Condition对象,这样就可以实现本方只唤醒对方的线程的操作。
 */
import java.util.concurrent.locks.*;

class Resource {
    private String name;
    private int count = 1;
    private boolean flag = false;
    private Lock lock = new ReentrantLock();
    private Condition condition = lock.newCondition();
    private Condition conditionPro = lock.newCondition();

    public void set(String name) throws InterruptedException {
        lock.lock();
        try {
            while (flag)
                condition.await();
            this.name = name + "--" + count++;
            System.out.println(Thread.currentThread().getName() + "...Producer..." + this.name);
            this.flag = true;
            conditionPro.signal();
        } finally {
            lock.unlock();    // 释放锁的动作一定要执行,否则会造成抛异常之后不释放锁,别的线程拿不到锁
        }
    }

    public void out() throws InterruptedException {
        lock.lock();
        try {
            while (!flag)
                conditionPro.await();
            System.out.println(Thread.currentThread().getName() + ".....Consumer....." + this.name);
            this.flag = false;
            condition.signal();
        } finally {
            lock.unlock();
        }
    }
}

class Producer implements Runnable {
    private Resource r;
    public Producer(Resource r) {
        this.r = r;
    }
    public void run() {
        while (true) {
            try {r.set("+Items+");} catch (InterruptedException e) {}
        }
    }
}

class Consumer implements Runnable {
    private Resource r;

    public Consumer(Resource r) {
        this.r = r;
    }

    public void run() {
        while (true) {
            try {r.out();} catch (InterruptedException e) {}
        }
    }
}


public class ProducerAndConsumer {
    public static void main(String[] args) {
        Resource r = new Resource();
        new Thread(new Producer(r)).start();
        new Thread(new Producer(r)).start();
        new Thread(new Consumer(r)).start();
        new Thread(new Consumer(r)).start();
    }
}

你可能感兴趣的:(java学习)