java并发--生产者消费者模型---volatile关键字

生产者消费者模型:系统中包含生产者和消费者两种角色,通过内存缓冲区进行通信。生产者生产消费者需要的产品,消费者把产品取出消费掉。
生产者===》 ====》消费者
生产者===》内存缓冲区(存放资源)====》消费者
生产者===》 ====》消费者
生产者消费者模型的实现:生产者是一堆线程,消费者是另一堆线程,内存缓冲区可以使用List数组队列,数据类型可以自定义。最关键就是内存缓冲区为空的时候消费者必须等待,而内存缓冲区满的时候,生产者必须等待。其他时候可以是个动态平衡。
多线程对临界区资源的操作时候必须保证在读写中只能存在一个线程,所以需要设计锁的策略。
volatile的变量是说这变量可能会被意想不到地改变,这样,编译器就不会去假设这个变量的值了。

public class Consumer implements Runnable {
    private BlockingQueue<DataModel> queue;
    private static final int SLEEPTIME = 1000;
    public Consumer(BlockingQueue<DataModel> queue){
        this.queue = queue;
    }
    @Override
    public void run() {
        System.out.println("消费者线程:"+Thread.currentThread().getId()+"开始消费");
        Random r = new Random();
        try{
            while(true){
                DataModel data = queue.take();
                if(data != null) {
                    int re = data.getData() * data.getData();
                    System.out.println(MessageFormat.format("{0}*{1}={2}", data.getData(),data.getData(),re));
                    Thread.sleep(r.nextInt(SLEEPTIME));
                }
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
            Thread.currentThread().interrupt();
        }
    }
}

/**
 * Created by dell on 2018/4/9.
 */
public class Producer implements Runnable {
    private volatile boolean flagCreate = true;
    //模拟线程池
    private BlockingDeque<DataModel> queue;

    //AtomicInteger这个类的存在是为了满足在高并发的情况下,原生的整形数值自增线程不安全的问题
    private static AtomicInteger count = new AtomicInteger();

    private static final int SLEEPTIME = 1000;

    public Producer(LinkedBlockingDeque<DataModel> queue) {
        this.queue = queue;
    }

    @Override
    public void run() {
        DataModel data = null;
        Random random = new Random();
        System.out.println("生产者线程" + Thread.currentThread().getId() + "开始生产");
        try {
            while (flagCreate) {
                data = new DataModel(count.incrementAndGet());
                System.out.println(data.toString() + "加入资源池");

                if (!queue.offer(data, 2, TimeUnit.SECONDS)) {
                    System.err.println(" 加入队列失败");
                }
            }
        }catch(InterruptedException e){
            e.printStackTrace();
            Thread.currentThread().interrupt();
        }
    }
    public void stop(){
        flagCreate = false;
    }
}

import java.util.ArrayList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.LinkedBlockingDeque;

/**
 * Created by dell on 2018/4/9.
 */
public class Main {
    public static void main(String[] args) throws InterruptedException {
        //基于链表的双端阻塞队列。
        LinkedBlockingDeque<DataModel> queue = new LinkedBlockingDeque<DataModel>(10);

        ArrayList<DataModel> arrayList = new ArrayList<>();
        Producer p1 = new Producer(queue);
        Producer p2 = new Producer(queue);
        Producer p3 = new Producer(queue);
        Consumer c1 = new Consumer(queue);
        Consumer c2 = new Consumer(queue);
        Consumer c3 = new Consumer(queue);


        ExecutorService service = Executors.newCachedThreadPool();
        service.execute(p1);
        service.execute(p2);
        service.execute(p3);
        service.execute(c1);
        service.execute(c2);
        service.execute(c3);


        p1.stop();
        p2.stop();
        p3.stop();

        service.shutdown();
        service.shutdownNow();
    }
}

/**
 * Created by dell on 2018/4/9.
 * 自定义的数据模型
 */
public class DataModel {
    private final int d;

    public DataModel(int d) {
        this.d = d;
    }

    public int getData() {
        return d;
    }

    @Override
    public String toString() {
        return "DataModel{" +
                "d=" + d +
                '}';
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值