Java高并发专题之13、JUC中的Condition

本文目标

  1. synchronized中实现线程等待和唤醒
  2. Condition简介及常用方法介绍及相关示例
  3. 使用Condition实现生产者消费者
  4. 使用Condition实现同步阻塞队列

Object对象中的wait(),notify()方法,用于线程等待和唤醒等待中的线程,大家应该比较熟悉,想再次了解的朋友可以移步到线程的基本操作

synchronized中等待和唤醒线程示例

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. /**
  4. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  5. */
  6. public class Demo1 {
  7. static Object lock = new Object();
  8. public static class T1 extends Thread {
  9. @Override
  10. public void run() {
  11. System.out.println(System.currentTimeMillis() + "," + this.getName() + "准备获取锁!");
  12. synchronized (lock) {
  13. System.out.println(System.currentTimeMillis() + "," + this.getName() + "获取锁成功!");
  14. try {
  15. lock.wait();
  16. } catch (InterruptedException e) {
  17. e.printStackTrace();
  18. }
  19. }
  20. System.out.println(System.currentTimeMillis() + "," + this.getName() + "释放锁成功!");
  21. }
  22. }
  23. public static class T2 extends Thread {
  24. @Override
  25. public void run() {
  26. System.out.println(System.currentTimeMillis() + "," + this.getName() + "准备获取锁!");
  27. synchronized (lock) {
  28. System.out.println(System.currentTimeMillis() + "," + this.getName() + "获取锁成功!");
  29. lock.notify();
  30. System.out.println(System.currentTimeMillis() + "," + this.getName() + " notify!");
  31. try {
  32. TimeUnit.SECONDS.sleep(5);
  33. } catch (InterruptedException e) {
  34. e.printStackTrace();
  35. }
  36. System.out.println(System.currentTimeMillis() + "," + this.getName() + "准备释放锁!");
  37. }
  38. System.out.println(System.currentTimeMillis() + "," + this.getName() + "释放锁成功!");
  39. }
  40. }
  41. public static void main(String[] args) throws InterruptedException {
  42. T1 t1 = new T1();
  43. t1.setName("t1");
  44. t1.start();
  45. TimeUnit.SECONDS.sleep(5);
  46. T2 t2 = new T2();
  47. t2.setName("t2");
  48. t2.start();
  49. }
  50. }

输出:

  1. 11563530109234,t1准备获取锁!
  2. 21563530109234,t1获取锁成功!
  3. 31563530114236,t2准备获取锁!
  4. 41563530114236,t2获取锁成功!
  5. 51563530114236,t2 notify!
  6. 61563530119237,t2准备释放锁!
  7. 71563530119237,t2释放锁成功!
  8. 81563530119237,t1释放锁成功!

代码结合输出的结果我们分析一下:

  1. 线程t1先获取锁,然后调用了wait()方法将线程置为等待状态,然后会释放lock的锁
  2. 主线程等待5秒之后,启动线程t2,t2获取到了锁,结果中1、3行时间相差5秒左右
  3. t2调用lock.notify()方法,准备将等待在lock上的线程t1唤醒,notify()方法之后又休眠了5秒,看一下输出的5、8可知,notify()方法之后,t1并不能立即被唤醒,需要等到t2将synchronized块执行完毕,释放锁之后,t1才被唤醒
  4. wait()方法和notify()方法必须放在同步块内调用(synchronized块内),否则会报错

Condition使用简介

在了解Condition之前,需要先了解一下重入锁ReentrantLock,可以移步到:JUC中的ReentranLock

任何一个java对象都天然继承于Object类,在线程间实现通信的往往会应用到Object的几个方法,比如wait()、wait(long timeout)、wait(long timeout, int nanos)与notify()、notifyAll()几个方法实现等待/通知机制,同样的, 在java Lock体系下依然会有同样的方法实现等待/通知机制。

从整体上来看Object的wait和notify/notify是与对象监视器配合完成线程间的等待/通知机制,而Condition与Lock配合完成等待通知机制,前者是java底层级别的,后者是语言级别的,具有更高的可控制性和扩展性。两者除了在使用方式上不同外,在功能特性上还是有很多的不同:

  1. Condition能够支持不响应中断,而通过使用Object方式不支持
  2. Condition能够支持多个等待队列(new 多个Condition对象),而Object方式只能支持一个
  3. Condition能够支持超时时间的设置,而Object不支持

Condition由ReentrantLock对象创建,并且可以同时创建多个,Condition接口在使用前必须先调用ReentrantLock的lock()方法获得锁,之后调用Condition接口的await()将释放锁,并且在该Condition上等待,直到有其他线程调用Condition的signal()方法唤醒线程,使用方式和wait()、notify()类似。

示例代码:

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class Demo2 {
  9. static ReentrantLock lock = new ReentrantLock();
  10. static Condition condition = lock.newCondition();
  11. public static class T1 extends Thread {
  12. @Override
  13. public void run() {
  14. System.out.println(System.currentTimeMillis() + "," + this.getName() + "准备获取锁!");
  15. lock.lock();
  16. try {
  17. System.out.println(System.currentTimeMillis() + "," + this.getName() + "获取锁成功!");
  18. condition.await();
  19. } catch (InterruptedException e) {
  20. e.printStackTrace();
  21. } finally {
  22. lock.unlock();
  23. }
  24. System.out.println(System.currentTimeMillis() + "," + this.getName() + "释放锁成功!");
  25. }
  26. }
  27. public static class T2 extends Thread {
  28. @Override
  29. public void run() {
  30. System.out.println(System.currentTimeMillis() + "," + this.getName() + "准备获取锁!");
  31. lock.lock();
  32. try {
  33. System.out.println(System.currentTimeMillis() + "," + this.getName() + "获取锁成功!");
  34. condition.signal();
  35. System.out.println(System.currentTimeMillis() + "," + this.getName() + " signal!");
  36. try {
  37. TimeUnit.SECONDS.sleep(5);
  38. } catch (InterruptedException e) {
  39. e.printStackTrace();
  40. }
  41. System.out.println(System.currentTimeMillis() + "," + this.getName() + "准备释放锁!");
  42. } finally {
  43. lock.unlock();
  44. }
  45. System.out.println(System.currentTimeMillis() + "," + this.getName() + "释放锁成功!");
  46. }
  47. }
  48. public static void main(String[] args) throws InterruptedException {
  49. T1 t1 = new T1();
  50. t1.setName("t1");
  51. t1.start();
  52. TimeUnit.SECONDS.sleep(5);
  53. T2 t2 = new T2();
  54. t2.setName("t2");
  55. t2.start();
  56. }
  57. }

输出:

  1. 1563532185827,t1准备获取锁!
  2. 1563532185827,t1获取锁成功!
  3. 1563532190829,t2准备获取锁!
  4. 1563532190829,t2获取锁成功!
  5. 1563532190829,t2 signal!
  6. 1563532195829,t2准备释放锁!
  7. 1563532195829,t2释放锁成功!
  8. 1563532195829,t1释放锁成功!

输出的结果和使用synchronized关键字的实例类似。

Condition.await()方法和Object.wait()方法类似,当使用Condition.await()方法时,需要先获取Condition对象关联的ReentrantLock的锁,在Condition.await()方法被调用时,当前线程会释放这个锁,并且当前线程会进行等待(处于阻塞状态)。在signal()方法被调用后,系统会从Condition对象的等待队列中唤醒一个线程,一旦线程被唤醒,被唤醒的线程会尝试重新获取锁,一旦获取成功,就可以继续执行了。因此,在signal被调用后,一般需要释放相关的锁,让给其他被唤醒的线程,让他可以继续执行。

Condition常用方法

Condition接口提供的常用方法有:

和Object中wait类似的方法

  1. void await() throws InterruptedException:当前线程进入等待状态,如果其他线程调用condition的signal或者signalAll方法并且当前线程获取Lock从await方法返回,如果在等待状态中被中断会抛出被中断异常;
  2. long awaitNanos(long nanosTimeout):当前线程进入等待状态直到被通知,中断或者超时
  3. boolean await(long time, TimeUnit unit) throws InterruptedException:同第二种,支持自定义时间单位,false:表示方法超时之后自动返回的,true:表示等待还未超时时,await方法就返回了(超时之前,被其他线程唤醒了)
  4. boolean awaitUntil(Date deadline) throws InterruptedException:当前线程进入等待状态直到被通知,中断或者到了某个时间
  5. void awaitUninterruptibly();:当前线程进入等待状态,不会响应线程中断操作,只能通过唤醒的方式让线程继续

和Object的notify/notifyAll类似的方法

  1. void signal():唤醒一个等待在condition上的线程,将该线程从等待队列中转移到同步队列中,如果在同步队列中能够竞争到Lock则可以从等待方法中返回。
  2. void signalAll():与1的区别在于能够唤醒所有等待在condition上的线程

Condition.await()过程中被打断

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class Demo4 {
  9. static ReentrantLock lock = new ReentrantLock();
  10. static Condition condition = lock.newCondition();
  11. public static class T1 extends Thread {
  12. @Override
  13. public void run() {
  14. lock.lock();
  15. try {
  16. condition.await();
  17. } catch (InterruptedException e) {
  18. System.out.println("中断标志:" + this.isInterrupted());
  19. e.printStackTrace();
  20. } finally {
  21. lock.unlock();
  22. }
  23. }
  24. }
  25. public static void main(String[] args) throws InterruptedException {
  26. T1 t1 = new T1();
  27. t1.setName("t1");
  28. t1.start();
  29. TimeUnit.SECONDS.sleep(2);
  30. //给t1线程发送中断信号
  31. System.out.println("1、t1中断标志:" + t1.isInterrupted());
  32. t1.interrupt();
  33. System.out.println("2、t1中断标志:" + t1.isInterrupted());
  34. }
  35. }

输出:

  1. 1t1中断标志:false
  2. 2t1中断标志:true
  3. 中断标志:false
  4. java.lang.InterruptedException
  5. at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.reportInterruptAfterWait(AbstractQueuedSynchronizer.java:2014)
  6. at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2048)
  7. at com.itsoku.chat09.Demo4$T1.run(Demo4.java:19)

调用condition.await()之后,线程进入阻塞中,调用t1.interrupt(),给t1线程发送中断信号,await()方法内部会检测到线程中断信号,然后触发InterruptedException异常,线程中断标志被清除。从输出结果中可以看出,线程t1中断标志的变换过程:false->true->false

await(long time, TimeUnit unit)超时之后自动返回

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class Demo5 {
  9. static ReentrantLock lock = new ReentrantLock();
  10. static Condition condition = lock.newCondition();
  11. public static class T1 extends Thread {
  12. @Override
  13. public void run() {
  14. lock.lock();
  15. try {
  16. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",start");
  17. boolean r = condition.await(2, TimeUnit.SECONDS);
  18. System.out.println(r);
  19. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",end");
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. } finally {
  23. lock.unlock();
  24. }
  25. }
  26. }
  27. public static void main(String[] args) throws InterruptedException {
  28. T1 t1 = new T1();
  29. t1.setName("t1");
  30. t1.start();
  31. }
  32. }

输出:

  1. 1563541624082,t1,start
  2. false
  3. 1563541626085,t1,end

t1线程等待2秒之后,自动返回继续执行,最后await方法返回false,await返回false表示超时之后自动返回

await(long time, TimeUnit unit)超时之前被唤醒

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class Demo6 {
  9. static ReentrantLock lock = new ReentrantLock();
  10. static Condition condition = lock.newCondition();
  11. public static class T1 extends Thread {
  12. @Override
  13. public void run() {
  14. lock.lock();
  15. try {
  16. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",start");
  17. boolean r = condition.await(5, TimeUnit.SECONDS);
  18. System.out.println(r);
  19. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",end");
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. } finally {
  23. lock.unlock();
  24. }
  25. }
  26. }
  27. public static void main(String[] args) throws InterruptedException {
  28. T1 t1 = new T1();
  29. t1.setName("t1");
  30. t1.start();
  31. //休眠1秒之后,唤醒t1线程
  32. TimeUnit.SECONDS.sleep(1);
  33. lock.lock();
  34. try {
  35. condition.signal();
  36. } finally {
  37. lock.unlock();
  38. }
  39. }
  40. }

输出:

  1. 1563542046046,t1,start
  2. true
  3. 1563542047048,t1,end

t1线程中调用condition.await(5, TimeUnit.SECONDS);方法会释放锁,等待5秒,主线程休眠1秒,然后获取锁,之后调用signal()方法唤醒t1,输出结果中发现await后过了1秒(1、3行输出结果的时间差),await方法就返回了,并且返回值是true。true表示await方法超时之前被其他线程唤醒了。

long awaitNanos(long nanosTimeout)超时返回

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class Demo7 {
  9. static ReentrantLock lock = new ReentrantLock();
  10. static Condition condition = lock.newCondition();
  11. public static class T1 extends Thread {
  12. @Override
  13. public void run() {
  14. lock.lock();
  15. try {
  16. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",start");
  17. long r = condition.awaitNanos(TimeUnit.SECONDS.toNanos(5));
  18. System.out.println(r);
  19. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",end");
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. } finally {
  23. lock.unlock();
  24. }
  25. }
  26. }
  27. public static void main(String[] args) throws InterruptedException {
  28. T1 t1 = new T1();
  29. t1.setName("t1");
  30. t1.start();
  31. }
  32. }

输出:

  1. 1563542547302,t1,start
  2. -258200
  3. 1563542552304,t1,end

awaitNanos参数为纳秒,可以调用TimeUnit中的一些方法将时间转换为纳秒。

t1调用await方法等待5秒超时返回,返回结果为负数,表示超时之后返回的。

waitNanos(long nanosTimeout)超时之前被唤醒

  1. package com.itsoku.chat09;
  2. import java.util.concurrent.TimeUnit;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class Demo8 {
  9. static ReentrantLock lock = new ReentrantLock();
  10. static Condition condition = lock.newCondition();
  11. public static class T1 extends Thread {
  12. @Override
  13. public void run() {
  14. lock.lock();
  15. try {
  16. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",start");
  17. long r = condition.awaitNanos(TimeUnit.SECONDS.toNanos(5));
  18. System.out.println(r);
  19. System.out.println(System.currentTimeMillis() + "," + this.getName() + ",end");
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. } finally {
  23. lock.unlock();
  24. }
  25. }
  26. }
  27. public static void main(String[] args) throws InterruptedException {
  28. T1 t1 = new T1();
  29. t1.setName("t1");
  30. t1.start();
  31. //休眠1秒之后,唤醒t1线程
  32. TimeUnit.SECONDS.sleep(1);
  33. lock.lock();
  34. try {
  35. condition.signal();
  36. } finally {
  37. lock.unlock();
  38. }
  39. }
  40. }

输出:

  1. 1563542915991,t1,start
  2. 3999988500
  3. 1563542916992,t1,end

t1中调用await休眠5秒,主线程休眠1秒之后,调用signal()唤醒线程t1,await方法返回正数,表示返回时距离超时时间还有多久,将近4秒,返回正数表示,线程在超时之前被唤醒了。

其他几个有参的await方法和无参的await方法一样,线程调用interrupt()方法时,这些方法都会触发InterruptedException异常,并且线程的中断标志会被清除。

同一个锁支持创建多个Condition

使用两个Condition来实现一个阻塞队列的例子:

  1. package com.itsoku.chat09;
  2. import java.util.LinkedList;
  3. import java.util.concurrent.locks.Condition;
  4. import java.util.concurrent.locks.ReentrantLock;
  5. /**
  6. * 微信公众号:路人甲Java,专注于java技术分享(带你玩转 爬虫、分布式事务、异步消息服务、任务调度、分库分表、大数据等),喜欢请关注!
  7. */
  8. public class BlockingQueueDemo<E> {
  9. int size;//阻塞队列最大容量
  10. ReentrantLock lock = new ReentrantLock();
  11. LinkedList<E> list = new LinkedList<>();//队列底层实现
  12. Condition notFull = lock.newCondition();//队列满时的等待条件
  13. Condition notEmpty = lock.newCondition();//队列空时的等待条件
  14. public BlockingQueueDemo(int size) {
  15. this.size = size;
  16. }
  17. public void enqueue(E e) throws InterruptedException {
  18. lock.lock();
  19. try {
  20. while (list.size() == size)//队列已满,在notFull条件上等待
  21. notFull.await();
  22. list.add(e);//入队:加入链表末尾
  23. System.out.println("入队:" + e);
  24. notEmpty.signal(); //通知在notEmpty条件上等待的线程
  25. } finally {
  26. lock.unlock();
  27. }
  28. }
  29. public E dequeue() throws InterruptedException {
  30. E e;
  31. lock.lock();
  32. try {
  33. while (list.size() == 0)//队列为空,在notEmpty条件上等待
  34. notEmpty.await();
  35. e = list.removeFirst();//出队:移除链表首元素
  36. System.out.println("出队:" + e);
  37. notFull.signal();//通知在notFull条件上等待的线程
  38. return e;
  39. } finally {
  40. lock.unlock();
  41. }
  42. }
  43. public static void main(String[] args) {
  44. BlockingQueueDemo<Integer> queue = new BlockingQueueDemo<>(2);
  45. for (int i = 0; i < 10; i++) {
  46. int data = i;
  47. new Thread(new Runnable() {
  48. @Override
  49. public void run() {
  50. try {
  51. queue.enqueue(data);
  52. } catch (InterruptedException e) {
  53. }
  54. }
  55. }).start();
  56. }
  57. for (int i = 0; i < 10; i++) {
  58. new Thread(new Runnable() {
  59. @Override
  60. public void run() {
  61. try {
  62. Integer data = queue.dequeue();
  63. } catch (InterruptedException e) {
  64. e.printStackTrace();
  65. }
  66. }
  67. }).start();
  68. }
  69. }
  70. }

代码非常容易理解,创建了一个阻塞队列,大小为3,队列满的时候,会被阻塞,等待其他线程去消费,队列中的元素被消费之后,会唤醒生产者,生产数据进入队列。上面代码将队列大小置为1,可以实现同步阻塞队列,生产1个元素之后,生产者会被阻塞,待消费者消费队列中的元素之后,生产者才能继续工作。

Object的监视器方法与Condition接口的对比

对比项 Object 监视器方法 Condition
前置条件 获取对象的锁 调用Lock.lock获取锁,调用Lock.newCondition()获取Condition对象
调用方式 直接调用,如:object.wait() 直接调用,如:condition.await()
等待队列个数 一个 多个,使用多个condition实现
当前线程释放锁并进入等待状态 支持 支持
当前线程释放锁进入等待状态中不响应中断 不支持 支持
当前线程释放锁并进入超时等待状态 支持 支持
当前线程释放锁并进入等待状态到将来某个时间 不支持 支持
唤醒等待队列中的一个线程 支持 支持
唤醒等待队列中的全部线程 支持 支持

总结

  1. 使用condition的步骤:创建condition对象,获取锁,然后调用condition的方法
  2. 一个ReentrantLock支持床多个condition对象
  3. void await() throws InterruptedException;方法会释放锁,让当前线程等待,支持唤醒,支持线程中断
  4. void awaitUninterruptibly();方法会释放锁,让当前线程等待,支持唤醒,不支持线程中断
  5. long awaitNanos(long nanosTimeout) throws InterruptedException;参数为纳秒,此方法会释放锁,让当前线程等待,支持唤醒,支持中断。超时之后返回的,结果为负数;超时之前返回的,结果为正数(表示返回时距离超时时间相差的纳秒数)
  6. boolean await(long time, TimeUnit unit) throws InterruptedException;方法会释放锁,让当前线程等待,支持唤醒,支持中断。超时之后返回的,结果为false;超时之前返回的,结果为true
  7. boolean awaitUntil(Date deadline) throws InterruptedException;参数表示超时的截止时间点,方法会释放锁,让当前线程等待,支持唤醒,支持中断。超时之后返回的,结果为false;超时之前返回的,结果为true
  8. void signal();会唤醒一个等待中的线程,然后被唤醒的线程会被加入同步队列,去尝试获取锁
  9. void signalAll();会唤醒所有等待中的线程,将所有等待中的线程加入同步队列,然后去尝试获取锁
来源:http://itsoku.com/course/1/13
posted @ 2022-05-04 19:41  程序员小明1024  阅读(103)  评论(0编辑  收藏  举报