Java之多线程开发时多条件Condition接口的使用

转:http://blog.csdn.net/a352193394/article/details/39454157

我们在多线程开发中,可能会出现这种情况。就是一个线程需要另外一个线程满足某某条件才能继续运行,或者需

要其他线程满足好几个条件才能运行,对于这样的多条件的多线程并发,我们如何控制好各个线程之间的关系,使他们

能很好的处理冲突不至于相互出现问题呢,下面我们来介绍一下Java提供的Condition这个接口,这个接口很好的实现了

这种需求。

对于这个问题最经典的例子就是生产者消费者模型,生产者当缓冲区满的时候不生产商品知道缓冲区有空余,消费

者当缓冲区为0 的时候不拿商品,直到生产者向缓冲区放入商品,下面我们使用Conditon这个接口来实现这样的需求。

[java] view plaincopyprint?

  1. import java.util.concurrent.ExecutorService;
  2. import java.util.concurrent.Executors;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.Lock;
  5. import java.util.concurrent.locks.ReentrantLock;
  6. /**
  7. * 有时候线程取得lock后需要在一定条件下才能做某些工作,比如说经典的Producer和Consumer问题。
  8. * 在Java 5.0以前,这种功能是由Object类的wait(), notify()和notifyAll()等方法实现的,
  9. * 在5.0里面,这些功能集中到了Condition这个接口来实现。
  10. */
  11. public class ConditionTest {
  12. /**
  13. * 篮子程序,这里为了简化问题,篮子中最多只能有一个苹果。
  14. * Consumer必须在篮子里有苹果的时候才能吃苹果,否则它必须暂时放弃对篮子的锁定,
  15. * 等到Producer往篮子里放了苹果后再去拿来吃。而Producer必须等到篮子空了才能往里放苹果,
  16. * 否则它也需要暂时解锁等Consumer把苹果吃了才能往篮子里放苹果。
  17. */
  18. public static class Basket {
  19. // 锁
  20. Lock lock = new ReentrantLock();
  21. //  根据锁产生Condition对象
  22. Condition produced = lock.newCondition();
  23. Condition consumed = lock.newCondition();
  24. // 篮子中的苹果数,最多为1
  25. int num = 0;
  26. /**
  27. * 生产苹果,往篮子里放
  28. * @throws InterruptedException
  29. */
  30. public void produce() throws InterruptedException {
  31. // 获得锁
  32. lock.lock();
  33. System.out.println("Producer get a lock...");
  34. try {
  35. // 判断是否满足生产条件
  36. while (num == 1) {
  37. // 如果有苹果,则不生产,放弃锁,进入睡眠
  38. // 等待消费者消费
  39. System.out.println("Producer sleep...");
  40. consumed.await();
  41. System.out.println("Producer awaked...");
  42. }
  43. /*生产苹果*/
  44. Thread.sleep(500);
  45. System.out.println("Producer produced an Apple.");
  46. num = 1;
  47. // 通知等待produced Condition的线程
  48. produced.signal();
  49. } finally {
  50. lock.unlock();
  51. }
  52. }
  53. /**
  54. * 消费苹果,从篮子中取
  55. * @throws InterruptedException
  56. */
  57. public void consume() throws InterruptedException {
  58. // 获得锁
  59. lock.lock();
  60. System.out.println("Consumer get a lock...");
  61. try {
  62. // 判断是否满足消费条件
  63. while (num == 0) {
  64. // 如果没有苹果,无法消费,则放弃锁,进入睡眠
  65. // 等待生产者生产苹果
  66. System.out.println("Consumer sleep...");
  67. produced.await();
  68. System.out.println("Consumer awaked...");
  69. }
  70. /*吃苹果*/
  71. Thread.sleep(500);
  72. System.out.println("Consumer consumed an Apple.");
  73. num = 0;
  74. // 发信号唤醒某个等待consumed Condition的线程
  75. consumed.signal();
  76. } finally {
  77. lock.unlock();
  78. }
  79. }
  80. }
  81. /**
  82. * 测试Basket程序
  83. */
  84. public static void testBasket() throws Exception {
  85. final Basket basket = new Basket();
  86. //  定义一个producer
  87. Runnable producer = new Runnable() {
  88. public void run() {
  89. try {
  90. basket.produce();
  91. } catch (InterruptedException ex) {
  92. ex.printStackTrace();
  93. }
  94. }
  95. };
  96. // 定义一个consumer
  97. Runnable consumer = new Runnable() {
  98. public void run() {
  99. try {
  100. basket.consume();
  101. } catch (InterruptedException ex) {
  102. ex.printStackTrace();
  103. }
  104. }
  105. };
  106. //  各产生3个consumer和producer
  107. ExecutorService service = Executors.newCachedThreadPool();
  108. for (int i = 0; i < 3; i++){
  109. service.submit(producer);
  110. }
  111. for (int i = 0; i < 3; i++){
  112. service.submit(consumer);
  113. }
  114. service.shutdown();
  115. }
  116. public static void main(String[] args) throws Exception {
  117. ConditionTest.testBasket();
  118. }
  119. }

 

时间: 2024-08-04 15:30:55

Java之多线程开发时多条件Condition接口的使用的相关文章

Java多线程开发之~~~多条件Condition接口的使用

我们在多线程开发中,可能会出现这种情况.就是一个线程需要另外一个线程满足某某条件才能继续运行,或者需 要其他线程满足好几个条件才能运行,对于这样的多条件的多线程并发,我们如何控制好各个线程之间的关系,使他们 能很好的处理冲突不至于相互出现问题呢,下面我们来介绍一下Java提供的Condition这个接口,这个接口很好的实现了 这种需求. 对于这个问题最经典的例子就是生产者消费者模型,生产者当缓冲区满的时候不生产商品知道缓冲区有空余,消费 者当缓冲区为0 的时候不拿商品,直到生产者向缓冲区放入商品

转载 多线程开发时线程局部变量的使用

多线程开发时线程局部变量的使用 http://blog.csdn.net/zsxxsz/article/details/6284759 2011-03-28 22:37197人阅读评论(0)收藏举报 一.概述 现在多核时代多线程开发越来越重要了,多线程相比于多进程有诸多优势(当然也有诸多劣势).在早期C的库中,有许多函数是线程不安全的,因为内 部用到了静态变量,比如:char *strtok(char *s, const char *delim); 该函数内部就有一个静态指针,如果多个线程同时调

Java Tread多线程(1)实现Runnable接口

作者 : 卿笃军 原文地址:http://blog.csdn.net/qingdujun/article/details/39347245 本文演示,Tread多线程实现Runnable接口,以及简单的说明为什么有这种创建线程的方法. 一.创建线程的2中方法: 1)继承Thread类实现多线程,参见我的上一篇文章:Java Tread多线程(0)一个简单的多线程实例 : 2)第二种方法就是实现Runnable接口,创建一个新线程. 二.为什么要有这两种方法创建线程呢? ①主要原因:就是方法1)不

多线程之线程通信条件Condition二

接上一篇,实现Condition三个条件,有这样一个应用: 1. 有三个进程,第一个进程运行1次,第二个进程运行2次,第三个进程运行3次: 2. 先运行第二个进程,然后第一个,然后第三个: 3.  依次运行5次循环. 分析: 此时若用Object的wait和notify是实现不了的,我们能够用Lock锁的Condition实现,我们须要定义三个信号条件,分别控制这三个进程. 实现例如以下: package andy.thread.test; import java.util.concurrent

多线程之线程通信条件Condition

Condition是Locks锁下的另一种线程通信之间唤醒.阻塞的实现.它下面的await,和signal能够实现Object下的wait,notify和notifyAll的全部功能,除此之外改监视器和已绑定到每个条件,可以实现多条件的监听.Condition实质是被绑定到一个锁上,腰围特定的Lock实例获得Condition,即用 newCondition()方法. Condition下的await()相对于Object下的wait(); 阻塞或中断之前状况,让其处于等待状态. Conditi

java中多线程中测试某个条件的变化用 if 还是用 while?

最近在研究wait和notify方法,发现有个地方要注意,但是网上又说得不是很明白的地方,就是经典的生产者和消费模式,使用wait和notify实现,判断list是否为空的这个为什么要用while而不能使用if呢?其实是因为当线程wait之后,又被唤醒的时候,是从wait后面开始执行,而不是又从头开始执行的,所以如果用if的话,被唤醒之后就不会在判断if中的条件,而是继续往下执行了,如果list只是添加了一个数据,而存在两个消费者被唤醒的话,就会出现溢出的问题了,因为不会在判断size是否==0

使用java做paypal开发时购买东西支付不成功的原因

没有设置网站习惯设定,登陆自己的paypal账户,在网站习惯设定上填写回调的url路径,这样就可以 支付成功了并且异步修改订单的状态. 支付成功后回调的url中获取不到任何参数 你没有在测试的卖家账户上设置PDT数据传输功能,所以获取不到任何的参数,例如价格.商品信息等等 获取到参数之后.为什么会出现链接超时 你应该把你的PDT的token复制过来.也就是传说中at参数.

sdk开发时,对外暴露的接口封装

思考,用同步还是异步? eg.登陆接口 1.定义接口LoginCallBack,两个函数 请求成功和失败 public interface LoginCallBack{ public void OnSuccess(SDKUser user); //登陆成功的回调,我们的需求是给调用者一个封装好的SDKUser类,一些用户信息的get/set 略 public void OnFailed(String msg); //传给调用者错误信息 } 2. 对外暴露的SDK主类中,进行处理具体业务逻辑 pu

sdk开发时,对外暴露的接口

思考,用同步还是异步? eg.登陆接口 1.定义接口LoginCallBack,两个函数 请求成功和失败 public interface LoginCallBack{ public void OnSuccess(SDKUser user); //登陆成功的回调,我们的需求是给调用者一个封装好的SDKUser类,一些用户信息的get/set 略 public void OnFailed(String msg); //传给调用者错误信息 } 2. 对外暴露的SDK主类中,进行处理具体业务逻辑 pu