- java提供了几个方法解决线程之间的通信问题
方法名 | 作用 |
---|---|
wait() | 表示线程一直等待,直到其他线程通知,与sleep不同,会释放锁 |
wait(long timeout) | 指定等待的毫秒数 |
notify | 唤醒一个处于等待状态的线程 |
notifyAll() | 唤醒同一个对象上所有调用wait()方法的线程,优先级别高的线程调度 |
注意:均是Object类的方法,都只能在同步方法或者同步代码块中使用,否则会抛出异常InterruptedException
管程法
- 生产者:负责生产数据的模块(可能是方法,对象,线程,进程);
- 消费者:负责处理数据的模块(可能是方法,对象,线程,进程);
- 缓冲区:消费者不能直接使用生产者的数据,他们之间有个”缓冲区“
生产者将生产好的数据放入缓冲区,消费者从缓冲区拿出数据
package com.zeng.thread;
//测试:生产者消费者模型-->利用缓冲区解决:管程法
//生产者,消费者,产品,缓冲区
public class TestPC {public static void main(String[] args) {SynContainer container=new SynContainer();new Productor(container).start();new Consumer(container).start();}
}//生产者
class Productor extends Thread{SynContainer container;public Productor(SynContainer container) {//构造器this.container=container;}//生产@Overridepublic void run() {for (int i = 0; i < 100; i++) {container.push(new Chicken(i));System.out.println("生产了"+i+"只鸡");}}
}
//消费者
class Consumer extends Thread{SynContainer container;public Consumer(SynContainer container){this.container=container;}@Overridepublic void run() {for (int i = 0; i < 100; i++) {System.out.println("消费了"+container.pop().id+"只鸡");}}
}//产品
class Chicken{int id;public Chicken(int id){this.id=id;}
}//缓冲区
class SynContainer{//需要一个容器大小Chicken[] chickens=new Chicken[10];//容器计数int count=0;//生产者放入产品public synchronized void push(Chicken chicken){//如果容器满了,就需要等待消费者消费if(count==chickens.length){//通知消费者消费,生产等待try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}chickens[count]=chicken;count++;//可以通知消费者消费了this.notifyAll();}//消费者消费产品public synchronized Chicken pop(){//判断能否消费if(count==0){//等待生产者生产,消费者等待try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}//如果可以消费count--;Chicken chicken=chickens[count];//吃完了,通知生产者生产this.notifyAll();return chicken;}
}
信号灯法
package com.zeng.thread;
//测试生产者消费者问题2:信号灯法,标志位解决
public class TestPC2 {public static void main(String[] args) {TV tv=new TV();new Player(tv).start();new Watcher(tv).start();}
}//生产者:演员
class Player extends Thread{TV tv;public Player(TV tv){this.tv=tv;}@Overridepublic void run() {for (int i = 0; i < 10; i++) {if(i%2==0){this.tv.play("披荆斩棘的哥哥播放中");}else{this.tv.play("明星大侦探");}}}
}//消费者:观众
class Watcher extends Thread{TV tv;public Watcher(TV tv){this.tv=tv;}@Overridepublic void run() {for (int i = 0; i < 10; i++) {tv.watch();}}
}//产品——>节目
class TV{//演员表演,观众等待T//观众观看,演员等待FString voice;//表演的节目boolean flag =true;//表演public synchronized void play(String voice){if(!flag){try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("演员表演了:"+voice);//通知观众观看this.notifyAll();//通知唤醒this.voice=voice;this.flag=!this.flag;}//观看public synchronized void watch(){if(flag){try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println("观看了:"+voice);//通知演员表演this.notifyAll();this.flag=!this.flag;}
}