使用wait、notify方法实现线程间的通信(注意这两个方法都是object的类的方法,换句话说java为所有的对象都提供了这两个方法)

1.wait和notify必须配合synchronized关键字使用

2.wait方法释放锁,notify方法不释放锁

示例1:

  1. public class ListAdd1 {
  2. private volatile static List list = new ArrayList();
  3.  
  4. public void add(){
  5. list.add("element");
  6. }
  7. public int size(){
  8. return list.size();
  9. }
  10.  
  11. public static void main(String[] args) {
  12.  
  13. final ListAdd1 list1 = new ListAdd1();
  14.  
  15. Thread t1 = new Thread(new Runnable() {
  16. public void run(){
  17. try {
  18. for(int i = 0; i <10; i++){
  19. list1.add();
  20. System.out.println("当前线程:" + Thread.currentThread().getName() + "添加了一个元素..");
  21. Thread.sleep(500);
  22. }
  23. } catch (InterruptedException e) {
  24. e.printStackTrace();
  25. }
  26. }
  27. }, "t1");
  28.  
  29. Thread t2 = new Thread(new Runnable() {
  30. public void run() {
  31. while(true){
  32. if(list1.size() == 5){
  33. System.out.println("当前线程收到通知:" + Thread.currentThread().getName() + " list size = 5 线程停止..");
  34. throw new RuntimeException();
  35. }
  36. }
  37. }
  38. }, "t2");
  39.  
  40. t1.start();
  41. t2.start();
  42. }
  43. }

上述代码实现了这样的功能:t1线程向一个list里面不断添加元素,t2线程死循环不断查看list的长度,当达到5的时候,t2程序退出。

以上程序有个不好的地方在于t2线程是不停的查看list的长度的,可以用其他的方式实现阻塞通知的效果就好了.

示例2:

  1. public class ListAdd2 {
  2.  
  3. private volatile static List list = new ArrayList();
  4.  
  5. public void add(){
  6. list.add("element");
  7. }
  8. public int size(){
  9. return list.size();
  10. }
  11.  
  12. public static void main(String[] args) {
  13.  
  14. final ListAdd2 list2 = new ListAdd2();
  15.  
  16. // 1 实例化出来一个 lock
  17. // 当使用wait 和 notify 的时候 , 一定要配合着synchronized关键字去使用
  18. final Object lock = new Object();
  19.  
  20. Thread t1 = new Thread(new Runnable() {
  21. public void run() {
  22. try {
  23. synchronized (lock) {
  24. for(int i = 0; i <10; i++){
  25. list2.add();
  26. System.out.println("当前线程:" + Thread.currentThread().getName() + "添加了一个元素..");
  27. Thread.sleep(500);
  28. if(list2.size() == 5){
  29. System.out.println("t1已经发出通知..");
  30. lock.notify();
  31. }
  32. }
  33. }
  34. } catch (InterruptedException e) {
  35. e.printStackTrace();
  36. }
  37.  
  38. }
  39. }, "t1");
  40.  
  41. Thread t2 = new Thread(new Runnable() {
  42. public void run() {
  43. synchronized (lock) {
  44. if(list2.size() != 5){
  45. try {
  46. lock.wait();
  47. } catch (InterruptedException e) {
  48. e.printStackTrace();
  49. }
  50. }
  51. System.out.println("当前线程:" + Thread.currentThread().getName() + "收到通知线程停止..");
  52. throw new RuntimeException();
  53. }
  54. }
  55. }, "t2");
  56.  
  57. t2.start();
  58. t1.start();
  59.  
  60. }
  61.  
  62. }

示例2执行的效果是:

  1. 当前线程:t1添加了一个元素..
  2. 当前线程:t1添加了一个元素..
  3. 当前线程:t1添加了一个元素..
  4. 当前线程:t1添加了一个元素..
  5. 当前线程:t1添加了一个元素..
  6. t1已经发出通知..
  7. 当前线程:t1添加了一个元素..
  8. 当前线程:t1添加了一个元素..
  9. 当前线程:t1添加了一个元素..
  10. 当前线程:t1添加了一个元素..
  11. 当前线程:t1添加了一个元素..
  12. 当前线程:t2收到通知线程停止..
  13. Exception in thread "t2" java.lang.RuntimeException
  14. at ListAdd2$.run(ListAdd2.java:)
  15. at java.lang.Thread.run(Thread.java:)

这里使用了wait和notify。首先注意到的是t2线程是先start的,然后t2线程获得lock,t2线程判断list的长度,长度这时候为0,t2调用了wait方法释放了lock。

t1线程启动后,lock被t2线程所持有,所以只能等待t2线程将锁释放。释放lock后,t1得到锁就开始向list中添加元素,当添加的元素的个数等于5的时候,调用notify方法,将t2线程唤醒。

但是t2线程被唤醒之后,并没有得到lock,只能等待t1线程先将锁释放,才能停下来。

示例2实现了示例1的基本功能,但是还不一样,因为在示例2中,t2的停止有很大的延迟,当list的长度等于5是,没有实时的停止下来。

示例3:

  1. public class ListAdd2 {
  2.  
  3. private volatile static List list = new ArrayList();
  4.  
  5. public void add() {
  6. list.add("element");
  7. }
  8.  
  9. public int size() {
  10. return list.size();
  11. }
  12.  
  13. public static void main(String[] args) {
  14. final ListAdd2 list2 = new ListAdd2();
  15. final CountDownLatch countDownLatch = new CountDownLatch(1);
  16. Thread t1 = new Thread(new Runnable() {
  17. public void run() {
  18. try {
  19. for (int i = 0; i < 10; i++) {
  20. list2.add();
  21. System.out.println("当前线程:" + Thread.currentThread().getName() + "添加了一个元素..");
  22. Thread.sleep(500);
  23. if (list2.size() == 5) {
  24. System.out.println("t1已经发出通知..");
  25. countDownLatch.countDown();
  26. }
  27. }
  28. } catch (InterruptedException e) {
  29. e.printStackTrace();
  30. }
  31.  
  32. }
  33. }, "t1");
  34.  
  35. Thread t2 = new Thread(new Runnable() {
  36. public void run() {
  37. if (list2.size() != 5) {
  38. try {
  39. countDownLatch.await();
  40. } catch (InterruptedException e) {
  41. e.printStackTrace();
  42. }
  43. }
  44. System.out.println("当前线程:" + Thread.currentThread().getName() + "收到通知线程停止..");
  45. throw new RuntimeException();
  46. }
  47. }, "t2");
  48.  
  49. t2.start();
  50. t1.start();
  51. }
  52. }

示例3使用CountDownLatch解决了实时性。

线程间通信--wait和notify的更多相关文章

  1. Java线程间通信之wait/notify

    Java中的wait/notify/notifyAll可用来实现线程间通信,是Object类的方法,这三个方法都是native方法,是平台相关的,常用来实现生产者/消费者模式.我们来看下相关定义: w ...

  2. Java 如何实现线程间通信?(notify、join、CountdownLatch、CyclicBarrier、FutureTask、Callable )

    转自:https://mp.weixin.qq.com/s?__biz=MzI4Njc5NjM1NQ==&mid=2247486499&idx=1&sn=d3f2d6959df ...

  3. (三)(1)线程间通信---wait和notify的使用

    这篇博客记录线程间通信相关api使用以及理解. 首先第一点,我之前的博客里的线程之间也是通信的,但是他们的通信是建立在访问的是同一个变量上的,相当于是变量.数据层面上的通信,而下面要讲的是线程层面上的 ...

  4. Java并发编程:线程间通信wait、notify

    Java并发编程:线程间协作的两种方式:wait.notify.notifyAll和Condition 在前面我们将了很多关于同步的问题,然而在现实中,需要线程之间的协作.比如说最经典的生产者-消费者 ...

  5. 线程间通信wait和notify【All】简介

    1.通信就是指相互交换一些数据或者发送一些控制指令,比如一个线程给另一个暂停执行的线程发送一个恢复执行的指令. 可变共享变量是天然的通信媒介,也就是说一个线程如果想和另一个线程通信的话,可以修改某个在 ...

  6. wait、notify、notifyAll实现线程间通信

    在Java中,可以通过配合调用Object对象的wait()方法和notify()方法或notifyAll()方法来实现线程间的通信.在线程中调用wait()方法,将阻塞等待其他线程的通知(其他线程调 ...

  7. java线程基础巩固---线程间通信快速入门,使用wait和notify进行线程间的数据通信

    之前已经对于线程同步相关的知识点进行了详细的学习,这次来学习一下线程间的通信相关的知识,话不多说直接用代码进行演练,以一个简陋的生产者消费者模型来初步了解下线程间通信是怎么一回事. 生产消费者第一版: ...

  8. 0038 Java学习笔记-多线程-传统线程间通信、Condition、阻塞队列、《疯狂Java讲义 第三版》进程间通信示例代码存在的一个问题

    调用同步锁的wait().notify().notifyAll()进行线程通信 看这个经典的存取款问题,要求两个线程存款,两个线程取款,账户里有余额的时候只能取款,没余额的时候只能存款,存取款金额相同 ...

  9. Java多线程编程核心技术---线程间通信(一)

    线程是操作系统中独立的个体,但这些个体如果不经过特殊处理就不能成为一个整体.线程间的通信就是成为整体的必用方案之一.线程间通信可以使系统之间的交互性更强大,在大大提高CPU利用率的同时还会使程序员对各 ...

随机推荐

  1. <转>lucene3.0 自学吧 四 termdocs

    http://www.cnblogs.com/LeftNotEasy/archive/2010/01/26/1656426.html http://www.doc100.net/bugs/t/5402 ...

  2. myBatis实例

    一.搭建环境, 建立数据库 CREATE TABLE user( id ) not NULL AUTO_INCREMENT, userName ) DEFAULT NULL, userAge ) DE ...

  3. types.MethodType

    http://stackoverflow.com/questions/972/adding-a-method-to-an-existing-object-instance 532down voteac ...

  4. 使用django表单,使网页添加上传文件,并分析文件。

    开发环境是: apache + python + django+ eclipse(开发环境) 欲达到目的: 在网页上,添加上传文件控件.然后读取csv文件,并分析csv文件. 操作步骤: django ...

  5. Appium+Maven+TestNG(ReportNG)环境搭建(详细过程)

    最近群里经常有人会问到关于maven构建Appium测试项目以及使用testNG生成测试报告的问题,试着搭建了一下,下面是过程: jdk安装过程我这里就不说了 一.下载eclipse,推荐下载Ecli ...

  6. TestCase--网站注册模块

    对于web测试,注册模块是必测的一个点,所以今天就来总结一下注册模块的测试用例 参考京东的注册页面 测试用例设计如下: 一.功能测试 1.  单击“免费注册”,页面是否正常跳转 2.  直接访问注册页 ...

  7. POJ1979 Red and Black (简单DFS)

    POJ1979 Description There is a rectangular room, covered with square tiles. Each tile is colored eit ...

  8. Gradle用户指南(章8:依赖关系管理基础)

    章8:依赖关系管理基础 本章将介绍一些gradle依赖关系管理的基础 什么是依赖关系管理? 简略的说,依赖管理是由两部分组成的.首先,gradle需要知道你要构建或者运行的项目,以便找到它们.我们将这 ...

  9. FastDFS的安装配置

    一:实验描述: fastdfs 介绍 FastDFS是一个开源的分布式文件系统,它对文件进行管理,功能包括:文件存储.文件同步.文件访问(文件上传.文件下载)等,解决了大容量存储和负载均衡的问题.特别 ...

  10. 学习总结 java基础