1、定义

即 Guarded Suspension,用在一个线程等待另一个线程的执行结果

要点

  • 有一个结果需要从一个线程传递到另一个线程,让他们关联同一个 GuardedObject

  • 如果有结果不断从一个线程到另一个线程那么可以使用消息队列

  • JDK 中,join 的实现、Future 的实现,采用的就是此模式

  • 因为要等待另一方的结果,因此归类到同步模式

2、实现

class GuardedObject {

   // 结果
   private Object response;
   private final Object lock = new Object();

   // 获取结果
   public Object get() {
       synchronized (lock) {
           // 条件不满足则等待
           while (response == null) {
               try {
                   lock.wait();
              } catch (InterruptedException e) {
                   e.printStackTrace();
              }
          }
           return response;
      }
  }

   // 产生结果
   public void complete(Object response) {
       synchronized (lock) {
           // 条件满足,通知等待线程
           this.response = response;
           lock.notifyAll();
      }
  }
}

3、应用

一个线程等待另一个线程的执行结果

public static void main(String[] args) {
   GuardedObject guardedObject = new GuardedObject();
   new Thread(() -> {
       try {
           // 子线程执行下载
           List<String> response = download(); // 模拟下载操作
           log.debug("download complete...");
           guardedObject.complete(response);
      } catch (IOException e) {
           e.printStackTrace();
      }
  }).start();

   log.debug("waiting...");
   // 主线程阻塞等待
   Object response = guardedObject.get();
   log.debug("get response: [{}] lines", ((List<String>) response).size());

}

执行结果

08:42:18.568 [main] c.TestGuardedObject - waiting...
08:42:23.312 [Thread-0] c.TestGuardedObject - download complete...
08:42:23.312 [main] c.TestGuardedObject - get response: [3] lines

4、带超时版 GuardedObject

如果要控制超时时间呢

class GuardedObjectV2 {

   private Object response;
   private final Object lock = new Object();

   public Object get(long millis) {
       synchronized (lock) {
           // 1) 记录最初时间
           long begin = System.currentTimeMillis();
           // 2) 已经经历的时间
           long timePassed = 0;
           while (response == null) {
               // 4) 假设 millis 是 1000,结果在 400 时唤醒了,那么还有 600 要等
               long waitTime = millis - timePassed;
               log.debug("waitTime: {}", waitTime);
               if (waitTime <= 0) {
                   log.debug("break...");
                   break;
              }
               try {
                   lock.wait(waitTime);  // 注意这里并不是 mills,防止虚假唤醒
              } catch (InterruptedException e) {
                   e.printStackTrace();
              }
               // 3) 如果提前被唤醒,这时已经经历的时间假设为 400
               timePassed = System.currentTimeMillis() - begin;
               log.debug("timePassed: {}, object is null {}",
                         timePassed, response == null);
          }
           return response;
      }
  }

   public void complete(Object response) {
       synchronized (lock) {
           // 条件满足,通知等待线程
           this.response = response;
           log.debug("notify...");
           lock.notifyAll();
      }
  }
}

测试,没有超时

public static void main(String[] args) {
   GuardedObjectV2 v2 = new GuardedObjectV2();
   new Thread(() -> {
       sleep(1); // 睡眠1秒
       v2.complete(null);
       sleep(1);
       v2.complete(Arrays.asList("a", "b", "c"));
  }).start();

   Object response = v2.get(2500);
   if (response != null) {
       log.debug("get response: [{}] lines", ((List<String>) response).size());
  } else {
       log.debug("can't get response");
  }
}

输出

08:49:39.917 [main] c.GuardedObjectV2 - waitTime: 2500
08:49:40.917 [Thread-0] c.GuardedObjectV2 - notify...
08:49:40.917 [main] c.GuardedObjectV2 - timePassed: 1003, object is null true
08:49:40.917 [main] c.GuardedObjectV2 - waitTime: 1497
08:49:41.918 [Thread-0] c.GuardedObjectV2 - notify...
08:49:41.918 [main] c.GuardedObjectV2 - timePassed: 2004, object is null false
08:49:41.918 [main] c.TestGuardedObjectV2 - get response: [3] lines

测试,超时

// 等待时间不足
List<String> lines = v2.get(1500);

输出

08:47:54.963 [main] c.GuardedObjectV2 - waitTime: 1500
08:47:55.963 [Thread-0] c.GuardedObjectV2 - notify...
08:47:55.963 [main] c.GuardedObjectV2 - timePassed: 1002, object is null true
08:47:55.963 [main] c.GuardedObjectV2 - waitTime: 498
08:47:56.461 [main] c.GuardedObjectV2 - timePassed: 1500, object is null true
08:47:56.461 [main] c.GuardedObjectV2 - waitTime: 0
08:47:56.461 [main] c.GuardedObjectV2 - break...
08:47:56.461 [main] c.TestGuardedObjectV2 - can't get response
08:47:56.963 [Thread-0] c.GuardedObjectV2 - notify...

5、多任务版 GuardedObject

图中 Futures 就好比居民楼一层的信箱(每个信箱有房间编号),左侧的 t0,t2,t4 就好比等待邮件的居民,右侧的 t1,t3,t5 就好比邮递员

如果需要在多个类之间使用 GuardedObject 对象,作为参数传递不是很方便,因此设计一个用来解耦的中间类,这样不仅能够解耦【结果等待者】和【结果生产者】,还能够同时支持多个任务的管理

新增 id 用来标识 Guarded Object

class GuardedObject {

   // 标识 Guarded Object
   private int id;

   public GuardedObject(int id) {
       this.id = id;
  }

   public int getId() {
       return id;
  }

   // 结果
   private Object response;

   // 获取结果
   // timeout 表示要等待多久 2000
   public Object get(long timeout) {
       synchronized (this) {
           // 开始时间 15:00:00
           long begin = System.currentTimeMillis();
           // 经历的时间
           long passedTime = 0;
           while (response == null) {
               // 这一轮循环应该等待的时间
               long waitTime = timeout - passedTime;
               // 经历的时间超过了最大等待时间时,退出循环
               if (timeout - passedTime <= 0) {
                   break;
              }
               try {
                   this.wait(waitTime); // 虚假唤醒 15:00:01
              } catch (InterruptedException e) {
                   e.printStackTrace();
              }
               // 求得经历时间
               passedTime = System.currentTimeMillis() - begin; // 15:00:02 1s
          }
           return response;
      }
  }

   // 产生结果
   public void complete(Object response) {
       synchronized (this) {
           // 给结果成员变量赋值
           this.response = response;
           this.notifyAll();
      }
  }
}

中间解耦类

class Mailboxes {
   private static Map<Integer, GuardedObject> boxes = new Hashtable<>();

   private static int id = 1;
   // 产生唯一 id
   private static synchronized int generateId() {
       return id++;
  }

   public static GuardedObject getGuardedObject(int id) {
       return boxes.remove(id);  // 注意这里的remove,防止堆溢出
  }

   public static GuardedObject createGuardedObject() {
       GuardedObject go = new GuardedObject(generateId());
       boxes.put(go.getId(), go);
       return go;
  }

   public static Set<Integer> getIds() {
       return boxes.keySet();
  }
}

业务相关类

class People extends Thread{
   @Override
   public void run() {
       // 收信
       GuardedObject guardedObject = Mailboxes.createGuardedObject();
       log.debug("开始收信 id:{}", guardedObject.getId());
       Object mail = guardedObject.get(5000);
       log.debug("收到信 id:{}, 内容:{}", guardedObject.getId(), mail);
  }
}
class Postman extends Thread {
   private int id;
   private String mail;

   public Postman(int id, String mail) {
       this.id = id;
       this.mail = mail;
  }

   @Override
   public void run() {
       GuardedObject guardedObject = Mailboxes.getGuardedObject(id);
       log.debug("送信 id:{}, 内容:{}", id, mail);
       guardedObject.complete(mail);
  }
}

测试

public static void main(String[] args) throws InterruptedException {
   for (int i = 0; i < 3; i++) {
       new People().start();
  }
   Sleeper.sleep(1);// 睡眠1秒
   for (Integer id : Mailboxes.getIds()) {
       new Postman(id, "内容" + id).start();
  }
}

某次运行结果

10:35:05.689 c.People [Thread-1] - 开始收信 id:3
10:35:05.689 c.People [Thread-2] - 开始收信 id:1
10:35:05.689 c.People [Thread-0] - 开始收信 id:2
10:35:06.688 c.Postman [Thread-4] - 送信 id:2, 内容:内容2
10:35:06.688 c.Postman [Thread-5] - 送信 id:1, 内容:内容1
10:35:06.688 c.People [Thread-0] - 收到信 id:2, 内容:内容2
10:35:06.688 c.People [Thread-2] - 收到信 id:1, 内容:内容1
10:35:06.688 c.Postman [Thread-3] - 送信 id:3, 内容:内容3
10:35:06.689 c.People [Thread-1] - 收到信 id:3, 内容:内容3

Java并发(二十三)----同步模式之保护性暂停的更多相关文章

  1. Java并发编程:同步容器

    Java并发编程:同步容器 为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch). ...

  2. 【转】Java并发编程:同步容器

    为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch).今天我们就来讨论下同步容器. ...

  3. 8、Java并发编程:同步容器

    Java并发编程:同步容器 为了方便编写出线程安全的程序,Java里面提供了一些线程安全类和并发工具,比如:同步容器.并发容器.阻塞队列.Synchronizer(比如CountDownLatch). ...

  4. java并发:线程同步机制之Volatile关键字&原子操作Atomic

    volatile关键字 volatile是一个特殊的修饰符,只有成员变量才能使用它,与Synchronized及ReentrantLock等提供的互斥相比,Synchronized保证了Synchro ...

  5. Java并发(十三):并发工具类——同步屏障CyclicBarrier

    先做总结 1.CyclicBarrier 是什么? CyclicBarrier 的字面意思是可循环使用(Cyclic)的屏障(Barrier).它要做的事情是,让一组线程到达一个屏障(也可以叫同步点) ...

  6. java并发:线程同步机制之Lock

    一.初识Lock Lock是一个接口,提供了无条件的.可轮询的.定时的.可中断的锁获取操作,所有加锁和解锁的方法都是显式的,其包路径是:java.util.concurrent.locks.Lock, ...

  7. Java并发(二):基础概念

    并发编程的第二部分,先来谈谈发布(Publish)与逸出(Escape); 发布是指:对象能够在当前作用域之外的代码中使用,例如:将对象的引用传递到其他类的方法中,对象的引用保存在其他类可以访问的地方 ...

  8. Java并发编程之同步

    1.synchronized 关键字 synchronized 锁什么?锁对象. 可能锁对象包括: this, 临界资源对象,Class 类对象. 1.1 同步方法 synchronized T me ...

  9. Java 并发编程 生产者消费者模式

    本文部分摘自<Java 并发编程的艺术> 模式概述 在线程的世界里,生产者就是生产数据的线程,消费者就是消费数据的数据.生产者和消费者彼此之间不直接通信,而是通过阻塞队列进行通信,所以生产 ...

  10. C#设计模式之二十三解释器模式(Interpreter Pattern)【行为型】

    一.引言   今天我们开始讲"行为型"设计模式的第十一个模式,也是面向对象设计模式的最后一个模式,先要说明一下,其实这个模式不是最后一个模式(按Gof的排序来讲),为什么把它放在最 ...

随机推荐

  1. 2021 VDC :vivo 互联网服务亿级用户的技术架构演进之路

    关注公众号[vivo互联网技术]--回复[2021VDC]获取大会PPT. 2021年12月16日,vivo 开发者大会圆满落幕.在互联网技术专场中,来自vivo 互联网技术的6位研发专家,从基础架构 ...

  2. 5分钟教会你如何在生产环境debug代码

    前言 有时出现的线上bug在测试环境死活都不能复现,靠review代码猜测bug出现的原因,然后盲改代码直接在线上测试明显不靠谱.这时我们就需要在生产环境中debug代码,快速找到bug的原因,然后将 ...

  3. Linux 文件目录配置及命令总结

    前言 在Linux中,一切皆文件,而每一个文件和目录都是从根目录开始的 Linux文件目录的作用 /bin目录:用来存放二进制可执行命令的目录,用户常用的命令都存在该目录下. /sbin目录:用来存放 ...

  4. python之HtmlTestRunner(三)中文字体乱码的情况

    使用HtmlTestRunner测试报告时,遇到中文字体无法识别的情况: 解决方案修改  \Lib\site-packages\HtmlTestRunner\result.py:def generat ...

  5. andriod sdk安装与使用

    一.进入以下网站下载 https://www.androiddevtools.cn/ 选择sdk工具-sdktools,这个工具比较好,可以通过SDK Manager下载到各种想要的包 有zip与ex ...

  6. Mysql有布尔(BOOL)类型吗

    转载请注明出处: 在MySQL中,没有专门的Boolean数据类型.相反,MySQL中使用TINYINT(1)来代表布尔类型,其中1表示真(True),0表示假(False).在MySQL中,TINY ...

  7. P2550 [AHOI2001] 彩票摇奖

    1.题目介绍 [AHOI2001] 彩票摇奖 题目描述 为了丰富人民群众的生活.支持某些社会公益事业,北塔市设置了一项彩票.该彩票的规则是: 每张彩票上印有 \(7\) 个各不相同的号码,且这些号码的 ...

  8. java - 正确关闭流

    package stream; import java.io.*; public class FileReaderTest { public static void main(String[] arg ...

  9. Laravel - Eloquent 删除数据

        public function ormDelete()     {         # 1.通过模型删除         // $student = Student::where('id',5 ...

  10. Oracle数据类型的简单学习之一

    Oracle数据类型的简单学习之一 背景 因为信创安可替代的发展 有很多项目提到了数据库切换到国产数据库的要求. 一般情况是要求从Oracle/SQLServer 迁移到国产的: 达梦/瀚高/人大金仓 ...