Java多线程编程——wait()和notify()、notifyAll()
1、源码
wait() notify() notifyAll()都是Object类中方法。源码如下所示:
public final native void notify();
public final native void notifyAll();
public final native void wait(long timeout) throws InterruptedException;
public final void wait() throws InterruptedException {
wait(0);
}
可以看到它们都是final native修饰的方法。
2、使用条件
首先我们需要明确的一点是wait() notify() notifyAll()都是需要在线程拥有对象锁的情况下使用。
在没有对象锁的情况下使用会抛出异常。执行下面代码,会抛出 java.lang.IllegalMonitorStateException 异常
package com.test;
public class Test {
public static void main(String[] args){
try {
Object obj = new Object();
obj.wait();
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}
}
}

3、作用
3.1 wait()方法的作用
wait()方法能够使当前线程停止,进入阻塞序列,等待被唤醒。
wait()方法被调用后,当前线程会立即释放持有的对象锁。其他线程可以通过竞争的方式获取该对象锁。
唤醒处于阻塞状态的线程,需要其他线程使用notify()或者notifyAll()方法。
3.2 notify()方法的作用
notify()方法能够唤醒处于阻塞状态(处于阻塞队列中)的线程。
需要注意的是调用一次notify()方法只能唤醒一个处于阻塞状态的线程。
而且调用notify()方法后,当前线程不会马上释放锁,被notify 的线程也不会马上获取该对象锁,而是需要等待当前线程执行完同步代码块。
3.3 notifyAll()方法的作用
notifyAll()方法能够唤醒正在等待同意共享资源的所有线程。
4、实例
直接上代码咯。
package com.test;
public class Test3 {
class WaitThread implements Runnable{
private Object lock;
public WaitThread(Object lock) {
// TODO Auto-generated constructor stub
super();
this.lock = lock;
}
@Override
public void run() {
// TODO Auto-generated method stub
try {
synchronized (lock) {
System.out.println(Thread.currentThread().getName() +
" 即将进入阻塞状态 wait time = " + System.currentTimeMillis());
lock.wait();
System.out.println(Thread.currentThread().getName() +
" 结束阻塞状态 wait time = " + System.currentTimeMillis());
}
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}
}
}
class NotifyThread implements Runnable{
private Object lock;
public NotifyThread(Object lock) {
// TODO Auto-generated constructor stub
super();
this.lock = lock;
}
@Override
public void run() {
// TODO Auto-generated method stub
try {
synchronized (lock) {
System.out.println(Thread.currentThread().getName() +
" 即将唤醒 notify time = " + System.currentTimeMillis());
lock.notify();
System.out.println(Thread.currentThread().getName() +
" 唤醒线程执行结束 notify time = " + System.currentTimeMillis());
}
} catch (Exception e) {
// TODO: handle exception
e.printStackTrace();
}
}
}
public static void main(String[] args){
try {
Object lock = new Object();
Test3 test3 = new Test3();
Thread waitThread = new Thread(test3.new WaitThread(lock));
waitThread.start();
Thread.sleep(3000);
Thread notifyThread = new Thread(test3.new NotifyThread(lock));
notifyThread.start();
} catch (Exception e) {
// TODO: handle exception
}
}
}

5、总结
1 wait() notify() notifyAll() 方法只能在线程拥有对象锁的时候调用
2 wait()方法调用后 当前线程会立即释放锁
3 notify() notifyAll()方法执行后 当前对象并不会立即释放锁,而是要等同步代码块执行完成后才能释放锁
4 notify()方法每次调用后只能唤醒处于阻塞队列中的一个线程,如果需要唤醒等待当前资源的所有线程需要使用notifyAll()方法
Java多线程编程——wait()和notify()、notifyAll()的更多相关文章
- java多线程的wait、notify/notifyAll区别
1.wait().notify/notifyAll() 方法是Object的本地final方法,无法被重写. 2.wait()使当前线程阻塞,前提是 必须先获得锁,一般配合synchronized ...
- Java多线程编程——进阶篇二
一.线程的交互 a.线程交互的基础知识 线程交互知识点需要从java.lang.Object的类的三个方法来学习: void notify() 唤醒在此对象监视器上等待的单个 ...
- Java多线程编程详解
转自:http://programming.iteye.com/blog/158568 线程的同步 由于同一进程的多个线程共享同一片存储空间,在带来方便的同时,也带来了访问冲突这个严重的问题.Ja ...
- Java多线程编程的常见陷阱(转)
Java多线程编程的常见陷阱 2009-06-16 13:48 killme2008 blogjava 字号:T | T 本文介绍了Java多线程编程中的常见陷阱,如在构造函数中启动线程,不完全的同步 ...
- Java多线程编程总结(精华)
Java多线程编程总结 2007-05-17 11:21:59 标签:多线程 java 原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http ...
- Java多线程编程核心技术
Java多线程编程核心技术 这本书有利于对Java多线程API的理解,但不容易从中总结规律. JDK文档 1. Thread类 部分源码: public class Thread implements ...
- 《Java 多线程编程核心技术》- 笔记
作为业务开发人员,能够在工作中用到的技术其实不多.虽然平时老是说什么,多线程,并发,注入,攻击!但是在实际工作中,这些东西不见得用得上.因为,我们用的框架已经把这些事做掉了. 比如web开发,外面有大 ...
- Java多线程编程核心技术(三)多线程通信
线程是操作系统中独立的个体,但这些个体如果不经过特殊的处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一,可以说,使线程间进行通信后,系统之间的交互性会更强大,在大大提高CPU利用率的同时 ...
- Java多线程编程实战指南(核心篇)读书笔记(三)
(尊重劳动成果,转载请注明出处:http://blog.csdn.net/qq_25827845/article/details/76686044冷血之心的博客) 博主准备恶补一番Java高并发编程相 ...
随机推荐
- 分块+二分 BZOJ 3343
3343: 教主的魔法 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1312 Solved: 585[Submit][Status][Discus ...
- vector的哈希值 Codecraft-17 and Codeforces Round #391 (Div. 1 + Div. 2, combined) C
http://codeforces.com/contest/757/problem/C 题目大意:有n个导管,每个体育馆有k种神奇宝贝,然后所有的n个体育馆中,一共有m中神奇宝贝.可知,每个神奇宝贝中 ...
- ⑤ 设计模式的艺术-05.原型(Prototype)模式
场景 思考一下:克隆技术是怎么样的过程? 克隆羊多利大家还记得吗? javascript语言中的,继承怎么实现?那里面也有prototype,大家还记得吗? 原型模式 通过new产生一个对象需要非常繁 ...
- .NET面试题系列(一)基本概念
什么是CLR CLR常用简写词语,CLR是公共语言运行库(Common Language Runtime)和Java虚拟机一样也是一个运行时环境,它负责资源管理(内存分配和垃圾收集等),并保证应用和底 ...
- 【BZOJ4565】【HAOI2016】字符合并 [状压DP][区间DP]
字符合并 Time Limit: 20 Sec Memory Limit: 256 MB[Submit][Status][Discuss] Description 有一个长度为 n 的 01 串,你 ...
- 脱离MVC使用Razor模板引擎
关于Razor模板引擎 1.简介 模板引擎:Razor.Nveocity.Vtemplate.Razor有VS自动提示.使用起来会方便一点. 但是Razor大多是在MVC下使用的. 那么如何在非MVC ...
- Redis数据类型之散列(hash)
1. 什么是散列 散列类似于一个字典,是一个<K, V>对的集合,不过这个key和value都只能是字符串类型的,不能嵌套,可以看做Java中的Map<String, String& ...
- python笔记之BytesIO
1. 什么是BytesIO BytesIO与StringIO类似,不同的是StringIO只能存放string,BytesIO是用来存放bytes的,它提供了在内存中读写字节的能力. 即在内存中读写字 ...
- virtualenv搭建虚拟环境
最近因为项目需要,要在CentOS 7 上搭建一套开发环境,虽说Python的背后有着庞大的开源社区支持,但是有一个缺点就是每个包的质量都参差不齐,如果我们在工作服务器上去测试安装每个包,就会造成整个 ...
- Angular 2.0 基础:服务
什么是服务(Service) 在Angular 2 中我们提到的服务 service 一般指的是 哪些能够被其他组件或者指令调用的 单一的,可共享的 代码块.当然,通过服务可以将数据和组件分开,这样就 ...