java 线程协作 wait(等待)与 notiy(通知)
一.wait()、notify()和notifyAll()
为了更好的支持多线程之间的协作,JDK提供了三个重要的本地方法
//调用某个对象的wait()方法能让当前线程阻塞,并且当前线程必须拥有此对象的锁.
public final void wait() throws InterruptedException {
wait(0);
}
//调用某个对象的notify()方法能够唤醒一个正在等待这个对象的锁的线程,如果有多个线程都在等待这个对象的锁,则只能唤醒其中一个线程
public final native void notify();
//notifyAll()方法能够唤醒所有正在等待这个对象锁的线程;
public final native void notifyAll();
如图:当一个拥有Object锁的线程调用 wait()方法时,就会使当前线程加入object.wait 等待队列中,并且释放当前占用的Object锁,这样其他线程就有机会获取这个Object锁,获得Object锁的线程调用notify()方法,就能在Object.wait 等待队列中随机唤醒一个线程(该唤醒是随机的与加入的顺序无关,优先级高的被唤醒概率会高),若果调用notifyAll()方法就唤醒全部的线程。注意:调用notify()方法后并不会立即释放object锁,会等待该线程执行完毕后释放Object锁。
代码:
public class WaitTest {
private static Object object=new Object();
public static void main(String[] args) {
Thread thread=new Thread(){
@Override
public void run() {
synchronized (object) {
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"进入启动");
try {
object.wait();//使当前线程进入等待(进入Object.wait队列)并释放对象锁
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"线程执行结束");
}
}
};
thread.start();
Thread thread_2=new Thread(){
@Override
public void run() {
synchronized (object) {
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"进入启动");
try {
object.notify();//随机在Object.waitd队列中唤醒一个正在等待该对象锁的线程
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"唤醒一个等待的线程");
Thread.sleep(10000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
};
thread_2.start();
}
}
执行结果:
1473306408730:Thread-0进入启动
1473306408731:Thread-1进入启动
1473306408731:Thread-1唤醒一个等待的线程
1473306418731:Thread-0线程执行结束
从时间戳中可以看出 Thread-1 在通知Thread-0 继续执行后,Thread-0 并未立即执行,而是等待Thread-1 释放Object锁,在重新获得Object锁后,才能继续执行。(最后两个时间戳相减刚好是10秒)
二、使用CountDownLatch让唤醒的线程立即执行
在上面的实例中我们实现了线程之间的通信,但线程Thread-0被唤醒后,并没有立即执行而是,等待Thread-1线程执行完毕释放锁后才执行,那我们如何做到让线程被唤醒后立即执行呢?在这里我们可以使用JDK1.5被引入的CountDownLatch类来轻松实现。
package com.jalja.org.thread.demo02; import java.util.concurrent.CountDownLatch; public class CountDownLatchTest {
public static void main(String[] args) {
final CountDownLatch countDownLatch=new CountDownLatch(2);
Thread thread=new Thread(){
@Override
public void run() {
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"进入启动");
try {
countDownLatch.await();//使当前线程进入等待(进入Object.wait队列)并释放对象锁
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"线程执行结束");
}
};
thread.start();
Thread thread_3=new Thread(){
@Override
public void run() {
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"进入启动");
try {
countDownLatch.countDown();//随机在Object.waitd队列中唤醒一个正在等待该对象锁的线程
countDownLatch.countDown();
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"唤醒一个等待的线程");
Thread.sleep(10000);
System.out.println(System.currentTimeMillis()+":"+Thread.currentThread().getName()+"线程结束");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
thread_3.start();
}
}
执行结果:
1500554407368:Thread-0进入启动
1500554407368:Thread-1进入启动
1500554407368:Thread-1唤醒一个等待的线程
1500554407369:Thread-0线程执行结束
1500554417369:Thread-1线程结束
由结果可以看出Thread-0在被唤醒后立即执行了(时间相差1毫秒)在10后,Thread-1 才执行结束。
java 线程协作 wait(等待)与 notiy(通知)的更多相关文章
- Java 线程间通信 —— 等待 / 通知机制
本文部分摘自<Java 并发编程的艺术> volatile 和 synchronize 关键字 每个处于运行状态的线程,如果仅仅是孤立地运行,那么它产生的作用很小,如果多个线程能够相互配合 ...
- JMM之Java线程间通讯——等待通知机制及其经典范式
在并发编程中,实际处理涉及两个关键问题:线程之间如何通信及线程之间如何同步(这里的线程是指并发执行的活动实体). 通信是指线程之间以何种机制来交换信息.在共享内存的并发模型里,线程之间共享程序的公共状 ...
- java 线程协作 join()
在实际开发中我们往往会遇到这样的情况一个线程的执行需要依赖另一个线程执行后的结果.即主线程生成并起动了子线程,如果子线程里要进行大量的耗时的运算,主线程往往将于子线程之前结束,但是如果主线程处理完其他 ...
- java 线程协作 yield()
yield():方法的定义 调用yield方法会让当前线程交出CPU权限,让CPU去执行其他的线程. 但是yield不能控制具体的交出CPU的时间,另外,yield方法只能让拥有相同优先级的线程有获取 ...
- Java 线程 —— Wait (等待)和 Notify(唤醒)
Wait (等待)和 Notify(唤醒) 这里讲了一个Wait (等待)和 Notfity(唤醒),下面这个实例(工厂,商店,消费者) 额,然后,你就知道了,需要写三个类:工厂类,Shop类,消费者 ...
- 一 java线程的等待/通知模型
java 中线程之间的通信问题,有这么一个模型:一个线程修改了一个对象的值,而另一个线程感知到了变化,然后进行相应的操作,整个过程开始于一个线程,而最终执行又是另一个线程.前者是生产者,后者就是消费者 ...
- 【Java并发专题之三】Java线程互斥、协作原理
(I)Java线程互斥原理之synchronized原理 从JDK5引入CAS原子操作,但没有对synchronized关键字做优化,而是增加了J.U.C.concurrent,concurrent包 ...
- Java线程与多线程教程
本文由 ImportNew - liken 翻译自 Journaldev. Java线程是执行某些任务的轻量级进程.Java通过Thread类提供多线程支持,应用可以创建并发执行的多个线程. 应用 ...
- 一步一步创建JAVA线程
(一)创建线程 要想明白线程机制,我们先从一些基本内容的概念下手. 线程和进程是两个完全不同的概念,进程是运行在自己的地址空间内的自包容的程序,而线程是在进程中的一个单一的顺序控制流,因此,单个进程可 ...
随机推荐
- Dojo框架学习笔记<二>
一.dojo/dom 该模块定义了Dojo Dom API,主要有以下几种用法: 1.dom.byId();(相当于document.getElementById()) ①最直接的用 ...
- Android 文章列表
Android --列表-- Android(1)-Handler Looper Message MessageQueuehttp://www.cnblogs.com/TS-qrt/articles ...
- multiple definition of `err_sys' 《UNIX环境高级编程》
本文地址:http://www.cnblogs.com/yhLinux/p/4079930.html 问题描述: [点击此处直接看解决方案] 在练习<UNIX环境高级编程>APUE程序清单 ...
- linux下 tar解压 gz解压 bz2等各种解压文件使用方法
http://alex09.iteye.com/blog/647128 大致总结了一下linux下各种格式的压缩包的压缩.解压方法. .tar 解包:tar xvf FileName.tar 打包:t ...
- Log4j写日志文件使用详解
Log4j输出到控制台成功,写入文件失败 - Log4j和commons log的整合 一.今天在使用commongs-logging.jar和log4j.properties来输出系统日志的时候,发 ...
- 基于Axure的快速原型方法
Axure是一个专业的快速原型设计工具,让负责定义需求和规格.设计功能和界面的专家能够快速创建应用软件或Web网站的线框图.流程图.原型和规格说明文档.作为专业的原型设计工具,它能快速.高效的创建原型 ...
- Effecvive Java读书笔记(一):创建和销毁对象
I.考虑静态工厂方法替代构造器 优势:1.有清晰的方法名称,方便调用:多参数构造器易出现调用错误 2.不必每次调用都创建新对象 3.可以返回原返回类型的任何子类型 4.创建参数化类型实例的时候,代码简 ...
- [置顶]PADS PCB功能使用技巧系列之NO.003- 如何统一修改元件标号字体?
LAYOUT完毕后进行元件标号字体调整时,你是否试图用Select Document+Select All来选定所有标号?可结果却并不令人满意. (1)在Layout中,选择菜单栏Edit -> ...
- 润乾报表之制作List列表
一般情况洗啊,如果sql查到多条数据,使用表格扩展的方式在一张表格里面(横展.纵展):目前的需求是,以报表为单位,做成List.例如,如果查出3条数据,预览的时候,要有3张格式相同内容有异的报表.如图 ...
- iPhone的Push(推送通知)功能原理浅析
第一部分:Push原理(以下绝大多数内容参考自.图片来自iPhone OS Reference Library)机制简介Push 的工作机制可以简单的概括为下图图中,Provider是指某个iPhon ...