成都科技网站建设费,合肥专业网站优化,万网域名注册登录,wordpress无法保存pages1 生产者消费者问题#x1f4cc; 线程通信应用的场景可以简单地描述为生产者和消费者问题假设仓库中只能存放一件产品#xff0c;生产者将生产出来的产品放入仓库#xff0c;消费者将仓库中产品取走消费#xff1b;如果仓库中没有产品#xff0c;则生产者将产品放入仓库 线程通信应用的场景可以简单地描述为生产者和消费者问题假设仓库中只能存放一件产品生产者将生产出来的产品放入仓库消费者将仓库中产品取走消费如果仓库中没有产品则生产者将产品放入仓库否则停止生产并等待直到仓库中的产品被消费者取走为止如果仓库中放有产品则消费者可以将产品取走消费否则停止消费并等待直到仓库中再次放入产品为止。 这是一个线程同步问题生产者和消费者共享同一个资源并且生产者和消费者之间相互依赖互为条件对于生产者没有生产产品之前要通知消费者等待而生产了产品之后又需要马上通知消费者消费对于消费者在消费之后要通知生产者已经结束消费需要生产新的产品以供消费 在生产者消费者问题中仅有synchronized是不够的synchronized可阻止并发更新同一个共享资源实现了同步synchronized不能用来实现不同线程之间的消息传递 通信2 线程通信2.1 线程方法方法名作用wait ()表示线程一直等待直到其他线程通知与 sleep 不同会释放锁wait (long timeOut)指定等待的毫秒数notify ()唤醒一个处于等待状态的线程notifyAll()唤醒同一个对象所有的调用 wait()方法的线程优先级高的优先调度2.2 管程法 要点生产者负责生产数据的模块可能是方法对象线程进程消费者负责处理数据的模块可能是方法对象线程进程缓冲区消费者不能直接使用生产者的数据他们之间有个“缓冲区 生产者将生产好的数据放入缓冲区消费者从缓冲区拿出数据 代码举例//需要生产者、消费者、产品、缓冲区
public class TestPC {public static void main(String[] args) {SynContainer container new SynContainer();new Productor(container).start();new Consumer(container).start();}}//产品-鸡
class Chicken{int id;public Chicken(int id){this.id id;}
}//生产者
class Productor extends Thread{SynContainer container;public Productor(SynContainer container){this.containercontainer;}Overridepublic void run() {for (int i 1; i 10; i) {container.pushTo(new Chicken(i));System.out.println(生产了第i只鸡);}}
}//消费者
class Consumer extends Thread{SynContainer container;public Consumer(SynContainer container){this.containercontainer;}Overridepublic void run() {for (int i 1; i 10; i) {System.out.println(消费了第container.popTo().id只鸡累积消费了第i只鸡);}}
}class SynContainer{/**定义一个容器的大小*/Chicken[] chickens new Chicken[5];/** 容器计数器*/int count;//生产者想容器中放入产品public synchronized void pushTo(Chicken chicken){//如果容器满了就需要等待消费者消费if(chickens.lengthcount){//通知消费者消费try {this.wait();}catch (InterruptedException e) {e.printStackTrace();}}//如果没有满就丢入产品chickens[count]chicken;count;//通知消费者消费this.notifyAll();}//消费者从中取出产品public synchronized Chicken popTo(){//判断能否消费if(count0){// 消费者等待生产者产出try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}//如果可以消费count--;Chicken chicken chickens[count];//吃完了通知生产者消费this.notifyAll();return chicken;}
}控制台输出一开始两个线程都会进容器不过消费者看到容器鸡的数量为0就会等待此时生产者进行生产当容器满了后生产者等待通知消费者来消费那么消费者进行消费通知生产者可以继续生产... 我把数量调小了大家可以把数量增加这样效果更显著。2.3 信号灯法简单来说就是利用标识控制线程等待和唤醒类似于信号灯红灯性绿灯停。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 5; i) {if (i%20){this.tv.play(电视节目i);}else {this.tv.play(电视广告i);}}}
}//消费者--观众
class Watcher extends Thread{TV tv;public Watcher(TV tv){this.tv tv;}Overridepublic void run() {for (int i 0; i 5; i) {this.tv.watch();}}
}//产品--节目
class TV{//演员表演观众等待-T观众观看演员等待-FString show;boolean flag true;//表演public synchronized void play(String show){if (!flag){try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println(演员表演了show);//通知观众观看this.notifyAll();this.show show;this.flag !flag;}//表演public synchronized void watch(){if (flag){try {this.wait();} catch (InterruptedException e) {e.printStackTrace();}}System.out.println(观众观看了show);//通知演员表演this.notifyAll();this.flag !flag;}
}控制台输出 可以看到表演和观看严格按照顺序来了2.4 线程池 要点背景经常创建和销毁、使用量特别大的资源比如并发情况下的线程对性能影响很大。思路提前创建好多个线程放入线程池中使用时直接获取使用完放回池中。可以避免频繁创建销毁、实现重复利用。类似生活中的公共交通工具。好处提高响应速度减少了创建新线程的时间降低资源消耗重复利用线程池中线程不需要每次都创建便于线程管理(corePoolSize核心池的大小、maximumPoolSize最大程数、keepAliveTime线程没有任务时最多保持多长时间后会终止) 线程池类JDK 5.0起提供了线程池相关API ExecutorService 和 ExecutorsExecutorService真正的线程池接口。常见子类ThreadPoolExecutorvoid execute(Runnable command)执行任务/命令没有返回值一般用来执行RunnableT FutureT submit(CallableT task)执行任务有返回值一般又来执行Callablevoid shutdown()关闭连接池Executors工具类、线程池的工厂类用于创建并返回不同类型的线程池 简单举例public class TestPool {public static void main(String[] args) {//1.创建线程池ExecutorService service Executors.newFixedThreadPool(5);service.execute(new Mythread());service.execute(new Mythread());service.execute(new Mythread());service.execute(new Mythread());//2.关闭连接service.shutdown();}
}class Mythread implements Runnable{Overridepublic void run() {for (int i 0; i 2; i) {System.out.println(Thread.currentThread().getName());}}
}控制台输出