首先我们来看一张图,这张图描述了线程操作的几个步骤。

图已经描述的很清楚了,这里除了wait()之外,其他的前面都已经接触过了。
这一章我们主要来说一下和wait()相关的操作,其实和wait()相关的还有另外两个函数notity()和notifyAll()。
这三个函数都是Object类里的方法
wait():使调用此方法的线程释放共享资源的锁,然后从运行状态退出,进入等待队列,直到被再次唤醒
notify():随机唤醒等待队列中等待同一个共享资源的“一个”线程,并使该线程退出等待队列,进入可运行状态,也就是说notofy()通知唤醒一个线程。
notifyAll():使所有正在等待队列中等待同一个共享资源的“全部”线程从等待状态退出,进入可运行状态。
注意:notify()和notifyAll()后并不是直接就释放锁的,而是等notify()后的代码执行完毕后才释放锁。而且这三个方法都必须在synchronize代码块里执行,否则会报错。
 
每个对象都有两个队列,一个是就绪队列,一个是阻塞队列。就绪队列存储了将要获得锁的线程,阻塞队列存储了所有被阻塞的线程。一个线程被唤醒后,才会进入就绪队列,等待CPU的调度,反之,一个线程被wait()之后,线程就进入了阻塞队列,等待下一次被唤醒。
 
线程会释放锁的三种情况:
1.线程任务执行完毕,释放锁。
2.线程调用wait()方法,释放锁。
3.线程执行过程中发生异常,释放锁。
 
生产-消费者模式:
单生产者,单消费者demo:
 
①.生产者:
  1. package com.multiThread.bean;
  2. publicclass P {
  3. privateObject lock;
  4. public P(Object lock){
  5. this.lock = lock;
  6. }
  7. publicvoidSetValue(){
  8. try{
  9. synchronized(lock){
  10. if(!ValueObject.value.equals("")){
  11. lock.wait();
  12. }
  13. String value =System.currentTimeMillis()+"_"+System.nanoTime();
  14. System.out.println("set值为:"+ value);
  15. ValueObject.value = value;
  16. lock.notify();
  17. }
  18. }catch(Exception e){
  19. e.printStackTrace();
  20. }
  21. }
  22. }
②.消费者
  1. package com.multiThread.bean;
  2. publicclass C {
  3. privateObject lock;
  4. public C(Object lock){
  5. this.lock = lock;
  6. }
  7. publicvoid getValue(){
  8. try{
  9. synchronized(lock){
  10. if(ValueObject.value.equals("")){
  11. lock.wait();
  12. }
  13. System.out.println("get的值:"+ValueObject.value);
  14. ValueObject.value ="";
  15. lock.notify();
  16. }
  17. }catch(Exception e){
  18. e.printStackTrace();
  19. }
  20. }
  21. }
③.生产者线程类
  1. package com.multiThread.thread;
  2. import com.multiThread.bean.P;
  3. publicclassThreadPimplementsRunnable{
  4. private P p;
  5. publicThreadP(P p){
  6. this.p = p;
  7. }
  8. @Override
  9. publicvoid run(){
  10. while(true){
  11. p.SetValue();
  12. }
  13. }
  14. }
④.消费者线程类
  1. package com.multiThread.thread;
  2. import com.multiThread.bean.C;
  3. publicclassThreadCimplementsRunnable{
  4. private C c;
  5. publicThreadC(C c){
  6. this.c = c;
  7. }
  8. @Override
  9. publicvoid run(){
  10. while(true){
  11. c.getValue();
  12. }
  13. }
  14. }
⑤.测试类
  1. package com.multiThread.test.productionConsumption;
  2. import com.multiThread.bean.C;
  3. import com.multiThread.bean.P;
  4. import com.multiThread.thread.ThreadC;
  5. import com.multiThread.thread.ThreadP;
  6. /**
  7. * 等待、通知机制
  8. */
  9. publicclassProduceCustomTest{
  10. publicstaticvoid main(String[] args){
  11. Object lock =newObject();
  12. P p =new P(lock);
  13. C c =new C(lock);
  14. ThreadP tp =newThreadP(p);
  15. ThreadC tc =newThreadC(c);
  16. Thread t1 =newThread(tp);
  17. Thread t2 =newThread(tc);
  18. t1.start();
  19. t2.start();
  20. }
  21. }
执行结果:
  1. set值为:1466176983137_22743411842275
  2. get的值:1466176983137_22743411842275
  3. set值为:1466176983137_22743412731096
  4. get的值:1466176983137_22743412731096
  5. set值为:1466176983137_22743412952232
  6. get的值:1466176983137_22743412952232
  7. set值为:1466176983137_22743413140860
  8. get的值:1466176983137_22743413140860
  9. .
  10. .
  11. .
可见生产者和消费者是交替执行的。
这里仅仅是单生产者和消费者。如果是多生产者和消费者会正常运行吗?
答案是否定的。因为notify()之后是随机唤醒一个线程,如果生产者唤醒的是生产者,那么就会一直处于wait(),造成死锁。
解决问题的方法:
    将notify()方法换成notifyAll()
 
 
 
 
 
 
 
 
 

多线程(五)~ wait/notify机制(等待/通知)的更多相关文章

  1. Java多线程学习(四)等待/通知(wait/notify)机制

    转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79690279 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...

  2. Java并发读书笔记:线程通信之等待通知机制

    目录 synchronized 与 volatile 等待/通知机制 等待 通知 面试常问的几个问题 sleep方法和wait方法的区别 关于放弃对象监视器 在并发编程中,保证线程同步,从而实现线程之 ...

  3. 超强图文|并发编程【等待/通知机制】就是这个feel~

    你有一个思想,我有一个思想,我们交换后,一个人就有两个思想 If you can NOT explain it simply, you do NOT understand it well enough ...

  4. 十六、Condition等待通知

    一.简介 我们可以使用syncronized和wait,notify实现等待通知.而syncronized的高级实现Lock,也可以实现等待通知,需要构造Condition的实例对象. JDK文档:h ...

  5. java 多线程:线程通信-等待通知机制wait和notify方法;(同步代码块synchronized和while循环相互嵌套的差异);管道通信:PipedInputStream;PipedOutputStream;PipedWriter; PipedReader

    1.等待通知机制: 等待通知机制的原理和厨师与服务员的关系很相似: 1,厨师做完一道菜的时间不确定,所以厨师将菜品放到"菜品传递台"上的时间不确定 2,服务员什么时候可以取到菜,必 ...

  6. java多线程系列(三)---等待通知机制

    等待通知机制 前言:本系列将从零开始讲解java多线程相关的技术,内容参考于<java多线程核心技术>与<java并发编程实战>等相关资料,希望站在巨人的肩膀上,再通过我的理解 ...

  7. Java Concurrency - wait & notify, 等待通知机制

    生产者消费者问题是一个常见的多线程同步案例:一组生产者线程和一组消费者线程共享一个初始状态为空.大小为 N 的缓冲区.只有当缓冲区没满的时候,生产者才能把消息放入缓冲区,否则必须等待:只有缓冲区不空的 ...

  8. Java多线程之三volatile与等待通知机制示例

    原子性,可见性与有序性 在多线程中,线程同步的时候一般需要考虑原子性,可见性与有序性 原子性 原子性定义:一个操作或者多个操作在执行过程中要么全部执行完成,要么全部都不执行,不存在执行一部分的情况. ...

  9. 《java多线程编程核心技术》不使用等待通知机制 实现线程间通信的 疑问分析

    不使用等待通知机制 实现线程间通信的 疑问分析 2018年04月03日 17:15:08       ayf 阅读数:33 编辑 <java多线程编程核心技术>一书第三章开头,有如下案例: ...

  10. java使用wait(),notify(),notifyAll()实现等待/通知机制

    public class WaitNotify { static boolean flag=true; static Object lock=new Object(); static class Wa ...

随机推荐

  1. bzoj1087 互不侵犯King 状压dp+bitset

    题目传送门 题目大意:中文题面. 思路:又是格子,n又只有9,所以肯定是状压dp,很明显上面一行的摆放位置会影响下一行,所以先预处理出怎样的二进制摆放法可以放在上下相邻的两行,这里推荐使用bitset ...

  2. SQL数据库查询一张表新建一个排序字段并根据某列的排序存储排序值

    现在有一张表如下Id Name Age Classify Score1 张一 18 一班 122 张二 17 二班 19 3 张三 19 三班 30 我跟据他们的分数进行排名 再去新建一个列存储排序值 ...

  3. vue修改组件样式

    .el-date-editor /deep/ input{ padding-left:30px; } 改变引入的组件里面元素的样式: 1.去掉css内的scoped,但是这样会污染全局 2.加上/de ...

  4. PIE SDK图像裁剪

    1.算法功能简介 图像裁剪的目的是获取选定的影像范围区域.图像裁切工具提供像素范围裁切.矢量裁切.栅格图像裁切和几何图元裁切四种方式. 像素范围裁切是基于像素坐标获取矩形裁切区域的裁切方式:矢量裁切是 ...

  5. vue中使用对非表单元素使用contenteditable的问题

    先说下问题,再上解决方案: span编辑时有多余空格和回车会影响样式(我用的是span便以此为例) 代码:(有换行符) 效果图: 代码:(无换行符) 效果图: 当在span标签的contentedit ...

  6. ajax请求方法及参数说明

    $.ajax()请求示例 <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> ...

  7. c语言3种方式实现参数传递

    学习计算机已经两年了,参数传递已经成功恶心了我两年,今天在写二叉树遍历的时候成功对此忍无可忍.本文是在阅读https://blog.csdn.net/u013097971/article/detail ...

  8. python 获取子目录下的所有文件的路径

    import os pathss=[] for root, dirs, files in os.walk(tarpath): path = [os.path.join(root, name) for ...

  9. keepalived+lvs简单实现

    一,部署实战环节 01,服务架构图: 服务器镜像为centos6.9   02,服务安装: 10.0.0.10: 下载keeplived官方包--> http://www.keepalived. ...

  10. sublime Text 3 官方版 3114 注册码

    —– BEGIN LICENSE —– Anthony Sansone Single User License EA7E-878563 28B9A648 42B99D8A F2E3E9E0 16DE0 ...