JAVA wait(), notify(),sleep具体解释

 
    在CSDN开了博客后,一直也没在上面公布过文章,直到前一段时间与一位前辈的对话,才发现技术博客的重要,立志要把CSDN的博客建好。但一直没有找到好的开篇的主题,今天再看JAVA线程相互排斥、同步的时候又有了新的体会,就以他作为开篇吧。

    在JAVA中,是没有类似于PV操作、进程相互排斥等相关的方法的。JAVA的进程同步是通过synchronized()来实现的,须要说明的是,JAVA的synchronized()方法类似于操作系统概念中的相互排斥内存块,在JAVA中的Object类型中,都是带有一个内存锁的,在有线程获取该内存锁后,其他线程无法訪问该内存,从而实现JAVA中简单的同步、相互排斥操作。明确这个原理,就能理解为什么synchronized(this)与synchronized(static XXX)的差别了,synchronized就是针对内存区块申请内存锁,thiskeyword代表类的一个对象,所以其内存锁是针对同样对象的相互排斥操作,而static成员属于类专有,其内存空间为该类全部成员共同拥有,这就导致synchronized()对static成员加锁,相当于对类加锁,也就是在该类的全部成员间实现相互排斥,在同一时间仅仅有一个线程可訪问该类的实例。假设仅仅是简单的想要实如今JAVA中的线程相互排斥,明确这些基本就已经够了。但假设须要在线程间相互唤醒的话就须要借助Object.wait(), Object.nofity()了。

    Obj.wait(),与Obj.notify()必需要与synchronized(Obj)一起使用,也就是wait,与notify是针对已经获取了Obj锁进行操作,从语法角度来说就是Obj.wait(),Obj.notify必须在synchronized(Obj){...}语句块内。从功能上来说wait就是说线程在获取对象锁后,主动释放对象锁,同一时候本线程休眠。直到有其他线程调用对象的notify()唤醒该线程,才干继续获取对象锁,并继续运行。对应的notify()就是对对象锁的唤醒操作。但有一点需要注意的是notify()调用后,并非立即就释放对象锁的,而是在对应的synchronized(){}语句块运行结束,自己主动释放锁后,JVM会在wait()对象锁的线程中随机选取一线程,赋予其对象锁,唤醒线程,继续运行。这样就提供了在线程间同步、唤醒的操作。Thread.sleep()与Object.wait()二者都能够暂停当前线程,释放CPU控制权,基本的差别在于Object.wait()在释放CPU同一时候,释放了对象锁的控制。

 

    单单在概念上理解清楚了还不够,须要在实际的样例中进行測试才干更好的理解。对Object.wait(),Object.notify()的应用最经典的样例,应该是三线程打印ABC的问题了吧,这是一道比較经典的面试题,题目要求例如以下:

建立三个线程,A线程打印10次A,B线程打印10次B,C线程打印10次C,要求线程同一时候执行,交替打印10次ABC。这个问题用Object的wait(),notify()就能够非常方便的解决。代码例如以下:

 

 

public class MyThreadPrinter2 implements Runnable {   private String name;   private Object prev;   private Object self;   private MyThreadPrinter2(String name, Object prev, Object self) {   this.name = name;   this.prev = prev;   this.self = self;   }   @Override  public void run() {   int count = 10;   while (count > 0) {   synchronized (prev) {   synchronized (self) {   System.out.print(name);   count--;  self.notify();   }   try {   prev.wait();   } catch (InterruptedException e) {   e.printStackTrace();   }   }   }   }   public static void main(String[] args) throws Exception {   Object a = new Object();   Object b = new Object();   Object c = new Object();   MyThreadPrinter2 pa = new MyThreadPrinter2("A", c, a);   MyThreadPrinter2 pb = new MyThreadPrinter2("B", a, b);   MyThreadPrinter2 pc = new MyThreadPrinter2("C", b, c);   new Thread(pa).start();new Thread(pb).start();new Thread(pc).start();    }   
}  


 

   
  
     先来解释一下其总体思路,从大的方向上来讲,该问题为三线程间的同步唤醒操作,基本的目的就是ThreadA->ThreadB->ThreadC->ThreadA循环运行三个线程。为了控制线程运行的顺序,那么就必需要确定唤醒、等待的顺序,所以每个线程必须同一时候持有两个对象锁,才干继续运行。一个对象锁是prev,就是前一个线程所持有的对象锁。另一个就是自身对象锁。基本的思想就是,为了控制运行的顺序,必需要先持有prev锁,也就前一个线程要释放自身对象锁,再去申请自身对象锁,两者兼备时打印,之后首先调用self.notify()释放自身对象锁,唤醒下一个等待线程,再调用prev.wait()释放prev对象锁,终止当前线程,等待循环结束后再次被唤醒。运行上述代码,能够发现三个线程循环打印ABC,共10次。程序运行的主要过程就是A线程最先运行,持有C,A对象锁,后释放A,C锁,唤醒B。线程B等待A锁,再申请B锁,后打印B,再释放B,A锁,唤醒C,线程C等待B锁,再申请C锁,后打印C,再释放C,B锁,唤醒A。看起来似乎没什么问题,但假设你细致想一下,就会发现有问题,就是初始条件,三个线程依照A,B,C的顺序来启动,依照前面的思考,A唤醒B,B唤醒C,C再唤醒A。可是这样的假设依赖于JVM中线程调度、运行的顺序。详细来说就是,在main主线程启动ThreadA后,需要在ThreadA运行完,在prev.wait()等待时,再切回线程启动ThreadB,ThreadB运行完,在prev.wait()等待时,再切回主线程,启动ThreadC,仅仅有JVM依照这个线程运行顺序运行,才干保证输出的结果是正确的。而这依赖于JVM的详细实现。考虑一种情况,例如以下:假设主线程在启动A后,运行A,过程中又切回主线程,启动了ThreadB,ThreadC,之后,因为A线程尚未释放self.notify,也就是B需要在synchronized(prev)处等待,而这时C却调用synchronized(prev)获取了对b的对象锁。这样,在A调用完后,同一时候ThreadB获取了prev也就是a的对象锁,ThreadC的运行条件就已经满足了,会打印C,之后释放c,及b的对象锁,这时ThreadB具备了运行条件,会打印B,也就是循环变成了ACBACB了。这样的情况,能够通过在run中主动释放CPU,来进行模拟。代码例如以下:

 

    public void run() {   int count = 10;   while (count > 0) {   synchronized (prev) {   synchronized (self) {   System.out.print(name);   count--;  try{Thread.sleep(1);}catch (InterruptedException e){e.printStackTrace();}self.notify();   }   try {   prev.wait();   } catch (InterruptedException e) {   e.printStackTrace();   }   }   }   }   


 


    执行后的打印结果就变成了ACBACB了。为了避免这样的与JVM调度有关的不确定性。须要让A,B,C三个线程以确定的顺序启动,终于代码例如以下:

 

  
public class MyThreadPrinter2 implements Runnable {   private String name;   private Object prev;   private Object self;   private MyThreadPrinter2(String name, Object prev, Object self) {   this.name = name;   this.prev = prev;   this.self = self;   }   @Override  public void run() {   int count = 10;   while (count > 0) {   synchronized (prev) {   synchronized (self) {   System.out.print(name);   count--;  try{Thread.sleep(1);}catch (InterruptedException e){e.printStackTrace();}self.notify();   }   try {   prev.wait();   } catch (InterruptedException e) {   e.printStackTrace();   }   }   }   }   public static void main(String[] args) throws Exception {   Object a = new Object();   Object b = new Object();   Object c = new Object();   MyThreadPrinter2 pa = new MyThreadPrinter2("A", c, a);   MyThreadPrinter2 pb = new MyThreadPrinter2("B", a, b);   MyThreadPrinter2 pc = new MyThreadPrinter2("C", b, c);   new Thread(pa).start();Thread.sleep(10);new Thread(pb).start();Thread.sleep(10);new Thread(pc).start();Thread.sleep(10);}   
}  


 

 

       这样才干够完美的解决该问题。通过这个样例也是想说明一下,非常多理论、概念如Obj.wait(),Obj.notify()等,理解起来,比較简单,可是在实际的应用其中,这里却是往往出现故障的地方。须要更加深入的理解。并在解决这个问题的过程中不断加深对概念的掌握。

 

                              ——欢迎转载,请注明出处 http://blog.csdn.net/zyplus——

 

转载于:https://www.cnblogs.com/zfyouxi/p/4516295.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/257274.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

通过键盘上下键 JS事件,控制候选词的选择项

效果图 JS代码 //上下键 选择事件 searchBackgroud 为样式,只做标记,无实质样式,因为和其他样式不兼容,只能添加CSS$(document).keydown(function (event) {var upDownClickNum $("#SearchTips .searchBackgroud ").l…

物理竞赛得奖学计算机,物理竞赛林紫琪带你探索清华学堂计算机科学实验班”(姚班)...

林紫琪,34届全国中学生物理竞赛全国第22名,获得女生最高分,入选国家集训队,现就读于清华姚班。这是一条小科普:“清华学堂计算机科学实验班”(姚班)由世界著名计算机科学家姚期智院士于2005年创办,致力于培…

Jmeter===Jmeter中使用CSV Data Set Config参数化不重复数据执行N遍(转)

Jmeter中使用CSV Data Set Config参数化不重复数据执行N遍 要求: 今天要测试上千条数据,且每条数据要求执行多次,(模拟多用户多次抽奖) 1.用户id有175个,且没有任何排序规则; 2.要求175个用户都…

[转]wireshark 实用过滤表达式(针对ip、协议、端口、长度和内容) 实例介绍

首先说几个最常用的关键字,“eq” 和 “”等同,可以使用 “and” 表示并且,“or”表示或者。“!" 和 "not” 都表示取反。 一、针对wireshark最常用的自然是针对IP地址的过滤。其中有几种情况: (1&#xff0…

[Flexbox] Using order to rearrange flexbox children

Using the order property we alter the order in which flexbox children appear on the page, without making changes to the dom. Desktop Mobile 转载于:https://www.cnblogs.com/Answer1215/p/5453671.html

计算机怎么更改用户头像像,Win10系统电脑账户头像怎么改成系统默认状态?

为了保护电脑的安全,我们可以设置登录账号密码,而账号的头像也是可以自行更换的。但是,某些时候,因为一些原因,我们需要将Win10系统账户的头像去掉,即改成默认状态。但是很多人都不清楚该怎么操作&#xff…

SPFA模板

今天去听2015ZJOI浙江省队第二试的集训,早上就是听得云里雾里的ORZ,下午某两集训队大神过来将题目,第一个进了IOI的我只听懂了10%ORZ,第二个人机交互很好玩,找个时间单独写下。 顺便附带膜拜各位聚聚,保我明…

LCM在Kernel中的代码分析

lcm的分析首先是mtkfb.c 1.mtk_init中platform_driver_register(&mtkfb_driver)注册平台驱动 panelmaster_init(); DBG_init(); mtkfb_ipo_init(); 2.mtkfb_probe进行普配 3.然后执行primary_display_init(mtkfb_find_lcm_driver(),lcd_fps); 4.mtkfb_find_lcm_driver()进行…

html ascii编码方式,HTML 字符集 参考手册

要正确显示一个 HTML 页面,浏览器必须知道要使用的字符集(字符编码)。HTML 字符集在 HTML 中,正确的字符编码是什么?HTML5 中默认的字符编码是 UTF-8。这并非总是如此。早期网络的字符编码是 ASCII 码。后来,从 HTML 2.0 到 HTML …

JavaScript 中的闭包和作用域链(读书笔记)

要想理解闭包,应当先理解JavaScript的作用域和作用域链。 JavaScript有一个特性被称之为“声明提前(hoisting)”,即JavaScript函数里声明的所有变量(但不涉及赋值)都被“提前”至函数体的顶部,“…

leetcode jump game ii

题目: Given an array of non-negative integers, you are initially positioned at the first index of the array. Each element in the array represents your maximum jump length at that position. Your goal is to reach the last index in the minimum numb…

韩师师范学院计算机科学与技术在哪个学区,2017年韩山师范学院本科插班生考试《数据结构》A卷...

韩山师范学院2017年本科插班生考试试卷计算机科学与技术 专业 数据结构 试卷(A 卷)一、单项选择题(每题2分,共30分)1. 对线性表,在下列哪种情况下应当采用链表表示?( ) A. 经常需要随机地存取元素 B. 经常需要进行插入和删除操作 C. 表中元素…

JAVA取随机数,石头剪刀布实例

一、取随机数: import java.util.Random; //导入随机数 public class Test{public static void main(String[] args){Random xx new Random(); //声明随机数int number xx.nextInt(10); //赋值随机数给numberSystem.out.println("随机数…

计算机网络犯罪和一般犯罪的不同,论计算机网络犯罪题稿.doc

目 录摘要2第一章、网络犯罪概念、特点以及构成特征5(一)网络犯罪的概念认定5(二)网络犯罪的特点6(三)网络犯罪的构成7第二章、?网络犯罪的类型9(一)网络色情和性骚扰9(二)欺诈9(三)贩卖、销售违禁物品11(四)妨害名誉、侵犯个人隐私12(五)?制造、传播计算机病毒12第三章、?网…

实例变量和静态变量(或类变量static)

一个类通过使用运算符new可以创建多个不同的对象,这些对象将被分配不同的内存空间,准确的说法是:不同对象的实例变量将被分配不同的内存空间,如果类中有类变量,那么所有对象的这个类变量都被分配到同一处内存&#xff…

DB2 数据库清表语句

truncate table DWDM2.tablename IMMEDIATE; alter table DWDM1.tablename activate not logged initially with empty table; but which one is best ? the truncate should be better 转载于:https://www.cnblogs.com/TendToBigData/p/10501485.html

cnblogs_504 Gateway Time-out

地址:http://zzk.cnblogs.com/s?tb&w%E6%B1%82%E8%81%8C 504 Gateway Time-out 504 Gateway Time-out The gateway did not receive a timely response from the upstream server or application. Sorry for the inconvenience. Please report this message an…

第一阶段

初步实现了相机的调用,做了简单界面,并没有实现核心功能 Button button (Button) findViewById(R.id.sao);button.setOnClickListener(new OnClickListener(){Overridepublic void onClick(View v) {Intent intent new Intent(MediaStore.ACTION_IMAGE…

JavaScript 详说事件机制之冒泡、捕获、传播、委托

DOM事件流(event flow )存在三个阶段:事件捕获阶段、处于目标阶段、事件冒泡阶段。 事件捕获(event capturing):通俗的理解就是,当鼠标点击或者触发dom事件时,浏览器会从根节点开始…

很棒的HTML5效果实例

2019独角兽企业重金招聘Python工程师标准>>> http://mrdoob.com/141/Internet_Explorer_with_WebGL 转载于:https://my.oschina.net/u/3647620/blog/1552495