一.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(通知)的更多相关文章

  1. Java 线程间通信 —— 等待 / 通知机制

    本文部分摘自<Java 并发编程的艺术> volatile 和 synchronize 关键字 每个处于运行状态的线程,如果仅仅是孤立地运行,那么它产生的作用很小,如果多个线程能够相互配合 ...

  2. JMM之Java线程间通讯——等待通知机制及其经典范式

    在并发编程中,实际处理涉及两个关键问题:线程之间如何通信及线程之间如何同步(这里的线程是指并发执行的活动实体). 通信是指线程之间以何种机制来交换信息.在共享内存的并发模型里,线程之间共享程序的公共状 ...

  3. java 线程协作 join()

    在实际开发中我们往往会遇到这样的情况一个线程的执行需要依赖另一个线程执行后的结果.即主线程生成并起动了子线程,如果子线程里要进行大量的耗时的运算,主线程往往将于子线程之前结束,但是如果主线程处理完其他 ...

  4. java 线程协作 yield()

    yield():方法的定义 调用yield方法会让当前线程交出CPU权限,让CPU去执行其他的线程. 但是yield不能控制具体的交出CPU的时间,另外,yield方法只能让拥有相同优先级的线程有获取 ...

  5. Java 线程 —— Wait (等待)和 Notify(唤醒)

    Wait (等待)和 Notify(唤醒) 这里讲了一个Wait (等待)和 Notfity(唤醒),下面这个实例(工厂,商店,消费者) 额,然后,你就知道了,需要写三个类:工厂类,Shop类,消费者 ...

  6. 一 java线程的等待/通知模型

    java 中线程之间的通信问题,有这么一个模型:一个线程修改了一个对象的值,而另一个线程感知到了变化,然后进行相应的操作,整个过程开始于一个线程,而最终执行又是另一个线程.前者是生产者,后者就是消费者 ...

  7. 【Java并发专题之三】Java线程互斥、协作原理

    (I)Java线程互斥原理之synchronized原理 从JDK5引入CAS原子操作,但没有对synchronized关键字做优化,而是增加了J.U.C.concurrent,concurrent包 ...

  8. Java线程与多线程教程

    本文由 ImportNew - liken 翻译自 Journaldev.   Java线程是执行某些任务的轻量级进程.Java通过Thread类提供多线程支持,应用可以创建并发执行的多个线程. 应用 ...

  9. 一步一步创建JAVA线程

    (一)创建线程 要想明白线程机制,我们先从一些基本内容的概念下手. 线程和进程是两个完全不同的概念,进程是运行在自己的地址空间内的自包容的程序,而线程是在进程中的一个单一的顺序控制流,因此,单个进程可 ...

随机推荐

  1. linux环境下安装mongodb

    最近有用到mongodb,顺便找到了以前的指南,顺便写一篇随笔,以后或许有用到的地方. 第一步:下载mongodb的linux版本,mongodb-linux-x86_64-3.2.4.tgz(去官网 ...

  2. Codevs 1021 (玛丽卡)

    题目描述 Description 麦克找了个新女朋友,玛丽卡对他非常恼火并伺机报复. 因为她和他们不住在同一个城市,因此她开始准备她的长途旅行. 在这个国家中每两个城市之间最多只有一条路相通,并且我们 ...

  3. 第十三章:降维:主成分分析PCA

  4. 如何给澳洲路局写信refound罚金,遇到交通罚款怎么办

    在澳洲,100%的司机收到过罚单,包括停车,超速,闯红灯等等,其罚金一般都在200-500之间,当然其单位是AUD.所以,对大多数留学生来说,收到罚金意味着一个礼拜要吃吐了. 本人就收到过一次超速罚单 ...

  5. xpath实例 --//span[contains(.,'资讯管理')]

    实际例子: 实例一: 不得不说下第一个例子,谢谢selenium论坛的大神门,XPATH还有这种方式的定位,传说是“治疗一切跌打损伤,及text找不到”(Antony群友帮解决的,在此谢谢)

  6. java的 new 关键字

    java的new关键字想必大家都知道这是实例化一个对象.没错,也是为新对象分配内存空间. 比如new MyDate(22,7,1964)这样一个案例,他的完成需要四部: 一.为新对象分配内存空间,将M ...

  7. 强类型DataSet的使用简明教程

    关于弱类型 DataSet的缺点: 无论何时从 DataSet检索值都是以Object类型返回,需要对它进行类型转换: 给其它开发者使用 时无法知道哪些列可用: 运行时才能知道所 有列名,数据绑定麻烦 ...

  8. phpcms v9 标签含义整理

    {template "content","header"}   ----------  调用根目录下phpcms\template\content\header ...

  9. C#基于Socket的简单聊天室实践

    序:实现一个基于Socket的简易的聊天室,实现的思路如下: 程序的结构:多个客户端+一个服务端,客户端都是向服务端发送消息,然后服务端转发给所有的客户端,这样形成一个简单的聊天室功能. 实现的细节: ...

  10. phoenix 开发API系列 目录

    phoenix 开发API系列(一)创建简单的http api phoenix 开发API系列(二)phoenix 各类 api 实现方式 phoenix 开发API系列(三)phoenix api ...