gpt4 book ai didi

基于线程的wait和notify使用,生产消费案例

转载 作者:qq735679552 更新时间:2022-09-27 22:32:09 25 4
gpt4 key购买 nike

CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.

这篇CFSDN的博客文章基于线程的wait和notify使用,生产消费案例由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.

多个线程可以相互竞争,也可以互相协作完成一件事情.

Object的相关方法 。

Object相关方法 描述
void wait() 让当前线程等待,如果没有被唤醒,就一直等待
void wait(long timeout) 让当前线程等待指定毫秒值,如果到了指定的毫秒值自动唤醒
void notify() 唤醒一个线程,唤醒的是当前对象锁下的一个线程
void notifyAll() 唤醒所有线程,唤醒的是当前对象锁下面的所有线程

这些方法一定要放在同步代码块中去使用,并且这些方法要通过锁对象去调用【***】 。

案例:

生产方每生产一个产品就需要等待(通知)消费方消费完产品后才能继续生产 。

消费方每消费一个产品就需要等待(通知)生产方去生产产品后才能继续消费.

案例图解

生产方逻辑图

基于线程的wait和notify使用,生产消费案例

消费方逻辑图

基于线程的wait和notify使用,生产消费案例

代码实现

【注意】 。

notify、wait写在同步代码块中,并且使用同一个对象(共有对象:仓库)进行操作.

this.cangku.wait() 和this.wait() 前一个使用的是仓库对象 ,后一个使用的是当前任务对象(使用后一个会造成死锁) 。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
//仓库 - 唯一(锁对象,任何对象都可以,用共有对象做锁对象)
class CangKu { //当作 锁对象
     //定义一个变量体现数量
     public int productNum = 0 ;
}
//生产方和消费方共用一个仓库
//生产方
class ProductTask implements Runnable {
     private CangKu cangKu; //共用一个仓库不能自己创建,由外部传入
     public ProductTask(CangKu cangKu) { //通过构造函数初始化
         this .cangKu = cangKu;
     }
     @Override
     public void run() {
         while ( true ) {
             //通知notify与等待wait必须写在同步代码块中
             synchronized ( this .cangKu) { //判断是否有锁可用,有就进入
                 if ( this .cangKu.productNum == 0 ) {
                     ++ this .cangKu.productNum;   //生产数目+1
                     System.out.println( "生产了一个产品,当前产品数目:" + this .cangKu.productNum);
                     //通知消费者,必须用同一个锁对象,不然会造成死锁
                     this .cangKu.notify();
                 } else {
                     //当前还有存货不用生产,等待通知
                     try {
                         System.out.println( "生产方等待中..." );
                         this .cangKu.wait();
                     } catch (InterruptedException e) {
                         e.printStackTrace();
                     }
                 } //end if
             } //end synchronized 出房间释放锁
         }
     }
}
//消费方
class ConsumerTask implements Runnable {
     private CangKu cangKu;
     public ConsumerTask(CangKu cangKu) {    //构造方法
         this .cangKu = cangKu;
     }
     @Override
     public void run() {
         while ( true ) {
             synchronized ( this .cangKu) {
                 //判断,仓库是否为0
                 if ( this .cangKu.productNum == 0 ) {
                     try {
                         System.out.println( "消费方等待中..." );
                         this .cangKu.wait();
                     } catch (InterruptedException e) {
                         e.printStackTrace();
                     }
                 } else {
                     //有货可以吃
                     -- this .cangKu.productNum ;
                     System.out.println( "消费了一个产品,当前产品数目:" + this .cangKu.productNum);
                     //通知生产方生产产品
                     this .cangKu.notify();
                 } //end if
             } //end synchronized
         }
     }
}
public class Wait_Notify_Demo {
     public static void main(String[] args) {
         //任务对象(生产方和消费方共用一个仓库)
         CangKu cangKu = new CangKu();
         ProductTask productTask = new ProductTask(cangKu);
         ConsumerTask consumerTask = new ConsumerTask(cangKu);
         //定义线程(用Executors线程池)
         ExecutorService pool = Executors.newFixedThreadPool( 2 );
         pool.submit(productTask);   //生产
         pool.submit(consumerTask);  //消费
     }
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持我.

原文链接:https://striveday.blog.csdn.net/article/details/109810336 。

最后此篇关于基于线程的wait和notify使用,生产消费案例的文章就讲到这里了,如果你想了解更多关于基于线程的wait和notify使用,生产消费案例的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com