前面的文章我们学习了可以使用wait()和notify()来进行线程间的同步。Java提供了高级API,同步队列来简化我们的使用。
同步队列定义:
在任何时刻,只允许一个线程插入或移除元素。 Java中提供了BlockingQueue接口,这个接口有大量的标准实现,常用的有下面几种:
1、linkedBlockingQueue:无界队列
2、ArrayBlockingQueue:固定大小的队列
3、synchronizedQueue:大小只有一个
BlockingQueue API
1、add()方法,往对列出新增元素,如果队列满了,抛出异常
2、put()方法,往队列新增元素,如果队列满了,进行blocking,不抛出异常
3、offer()方法,往队列新增元素,如果成功返回true,插入失败返回false
4、take()方法,从队列中获取元素,如果队列为null,它会进行block等着有元素,如果有头元素,则取出它并移除。
5、poll()方法,从对了中获取元素,如果队列为null,返回null, 如果头元素存在,则取出被移除
生产者-消费者使用同步队列的例子
下面我们模拟生产者、消费者的场景:4个生产者线程同步的生产100内的数字,而N个消费者线程消费队列的数字。在队列加入了用于退出的数字。我们看下面的代码:
生产者代码:
![cadacd757e39e81de087af730424886c.png](https://i-blog.csdnimg.cn/blog_migrate/44a954f1af8069a2ef8e99061f039b99.jpeg)
消费者代码:
![dbb047ed1e7d25f21a052a71c4898820.png](https://i-blog.csdnimg.cn/blog_migrate/3a15478bfadd819f47bb72b05d1dc236.jpeg)
输出结果:
![97f1c50661a6b1a152257f65c8c5c49e.png](https://i-blog.csdnimg.cn/blog_migrate/31d64cecfc1d312f46f3dc54f3adcf9a.jpeg)
程序分析:
1、4个生产者同步的生产100内的数字,使用LinkedBlockingQueue.put()方法往队列增加,没有加锁就可以实现自动的同步
2、4个消费者通过take() 消费LinkedBlockingQueue队列的数据,如果没有拿到数据就阻塞,直到有数据
3、从输出中,我们发现使用同步队列比使用wait()、Notify()要简单很多。