shell
#回顾
1.在java编程语言中,与线程相关的方法主要有:
1.1.Object.wait/Object.notify/Object/notifyAll
1.2.Thread.sleep/Thread.join/Thread.yield
2.需要把各个方法的作用和使用场景说清楚,需要用两篇文章进行分享
#考考你
1.你知道wait/notify/notifyAll方法的作用吗?
2.你知道notify与notifyAll方法的区别吗?
3.你知道sleep方法的作用吗?
4.你知道wait与sleep方法的区别吗?
5.你知道join方法的作用吗?
6.你知道yield方法的作用吗?
案例:wait/notify/notifyAll
方法简述:
wait:释放线程拥有的当前锁对象,让线程进入WAITING状态
notify:唤醒当前锁对象等待池(WaitSet)中,某一个线程
notifyAll:唤醒当前锁对象等待池(WaitSet)中,所有线程
业务描述:
1.通过wait与notify(notifyAll)实现线程协同,实现经典的生产者消费者模式
2.编写生产者,向队列中生产数据,如果队列满,生产者进行等待,并唤醒消费者进行消费
3.编写消费者,从队列中消费数据,如果队列空,消费者进行等待,并唤醒生产者进行生产
4.在主类main方法中,创建两个生产者线程进行生产
5.在主类main方法中,创建一个消费者线程进行消费
生产者:
java
/**
* 生产者
*/
class Producer implements Runnable{
public void run() {
while(true){
synchronized (ProducerConsumerDemo.LOCK){
// 检查队列是否满,推荐用while而不是if
while(ProducerConsumerDemo.QUEUE.size() ==
ProducerConsumerDemo.CAPACITY){
// 如果队列满,则等待消费者消费
try {
System.out.println("队列满size:" +ProducerConsumerDemo.QUEUE.size()+ ",线程【" +
Thread.currentThread().getName() + "】正在等待消费者消费.");
ProducerConsumerDemo.LOCK.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 如果队列不满,则正常生产
ProducerConsumerDemo.QUEUE.add(1);
System.out.println("线程【" + Thread.currentThread().getName() +
"】生产了数据.当前队列size:" + ProducerConsumerDemo.QUEUE.size());
// 唤醒消费者
ProducerConsumerDemo.LOCK.notifyAll();
}
}
}
}
消费者:
java
/**
* 消费者
*/
class Consumer implements Runnable{
public void run() {
while(true){
synchronized (ProducerConsumerDemo.LOCK){
// 检查队列是否空,推荐用while而不是if
while(ProducerConsumerDemo.QUEUE.isEmpty()){
// 如果队列空,则等待生产者生产
try {
System.out.println("队列空size:" +ProducerConsumerDemo.QUEUE.size()+ ",线程【" +
Thread.currentThread().getName() + "】正在等待生产者生产.");
ProducerConsumerDemo.LOCK.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 如果队列不空,则正常消费
ProducerConsumerDemo.QUEUE.remove(0);
System.out.println("线程【" + Thread.currentThread().getName() +
"】消费了数据.当前队列size:" + ProducerConsumerDemo.QUEUE.size());
// 唤醒生产者
ProducerConsumerDemo.LOCK.notifyAll();
}
}
}
}
完整代码:
java
package com.anan.thread.threadmethod;
import java.util.ArrayList;
/**
* 生产者消费者模型:wait与notify(notifyAll)
*/
public class ProducerConsumerDemo {
// 定义公共资源:队列容量、队列
public final static Integer CAPACITY = 6;
public final static ArrayList<Integer> QUEUE = new ArrayList<Integer>(CAPACITY);
// 定义锁对象
public final static Object LOCK = new Object();
public static void main(String[] args) {
// 创建两个生产者线程
Runnable r1 = new Producer();
Thread producer0 = new Thread(r1,"producer-0");
producer0.start();
Thread producer1 = new Thread(r1,"producer-1");
producer1.start();
// 创建消费者线程
Runnable r2 = new Consumer();
Thread consumer1 = new Thread(r2,"consumer-0");
consumer1.start();
}
}
/**
* 生产者
*/
class Producer implements Runnable{
public void run() {
while(true){
synchronized (ProducerConsumerDemo.LOCK){
// 检查队列是否满,推荐用while而不是if
while(ProducerConsumerDemo.QUEUE.size() ==
ProducerConsumerDemo.CAPACITY){
// 如果队列满,则等待消费者消费
try {
System.out.println("队列满size:" +ProducerConsumerDemo.QUEUE.size()+ ",线程【" +
Thread.currentThread().getName() + "】正在等待消费者消费.");
ProducerConsumerDemo.LOCK.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 如果队列不满,则正常生产
ProducerConsumerDemo.QUEUE.add(1);
System.out.println("线程【" + Thread.currentThread().getName() +
"】生产了数据.当前队列size:" + ProducerConsumerDemo.QUEUE.size());
// 唤醒消费者
ProducerConsumerDemo.LOCK.notifyAll();
}
}
}
}
/**
* 消费者
*/
class Consumer implements Runnable{
public void run() {
while(true){
synchronized (ProducerConsumerDemo.LOCK){
// 检查队列是否空,推荐用while而不是if
while(ProducerConsumerDemo.QUEUE.isEmpty()){
// 如果队列空,则等待生产者生产
try {
System.out.println("队列空size:" +ProducerConsumerDemo.QUEUE.size()+ ",线程【" +
Thread.currentThread().getName() + "】正在等待生产者生产.");
ProducerConsumerDemo.LOCK.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 如果队列不空,则正常消费
ProducerConsumerDemo.QUEUE.remove(0);
System.out.println("线程【" + Thread.currentThread().getName() +
"】消费了数据.当前队列size:" + ProducerConsumerDemo.QUEUE.size());
// 唤醒生产者
ProducerConsumerDemo.LOCK.notifyAll();
}
}
}
}
总结
shell
#考考你答案
1.你知道wait/notify/notifyAll方法的作用吗?
1.1.wait/notify/notifyAll都是Object中的方法
1.2.通过等待与唤醒,实现线程之间的协同
1.3.wait方法会释放当前锁对象,让线程进入WAITING状态
1.4.notify方法用于:
1.4.1.唤醒当前锁对象等待池(WaitSet)中,正在等待
当前锁对象的某一个线程
1.4.2.让该线程进入RUNNABLE状态,并移入锁池(EntrySet)中,
重新竞争锁对象
1.5.notifyAll方法用于:
1.5.1.唤醒当前锁对象等待池(WaitSet)中,正在等待
当前锁对象的所有线程
1.5.2.让等待当前锁对象的所有线程,进入RUNNABLE状态,并
移入锁池(EntrySet)中,重新竞争锁对象
2.你知道notify与notifyAll方法的区别吗?
2.1.通过以上1.4点、1.5点已经说明了notify
与notifyAll的区别
2.2.这里可能有朋友不明白锁池(EntrySet),与
等待池(WaitSet)的概念。我们通过一个图进行说明:
#流程文字描述:
1.锁池(EntrySet):代表正在等待同一锁对象的线程集合,线程进入BLOCKED状态
2.拥有者(Owner):代表已经获取到锁对象的线程
3.等待池(WaitSet):代表已经进入WAITING状态,等待被唤醒的线程集合
4.流程描述:
4.1.enter:线程准备获取锁对象,此时锁被其它线程占有,线程进入EntrySet
4.2.acquire:当其它线程释放锁对象,EntrySet中的某个线程获取占有锁对象
4.3.release:占有锁对象线程,通过wait方式释放锁对象,进入WaitSet中,等待被唤醒
4.4.acquire:当有其它线程,通过notify/notifyAll方法唤醒WaitSet中线程,线程将重新进入EntrySet,重新竞争获取锁对象
4.5.release and exit:占有锁对象线程,释放锁并退出执行,线程生命周期结束