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. java调用本机的命令 如ping、打开文本等

    最近接触到用java代码调用主机的命令部分感觉有点意思整理总结一下 环境jdk1.8  操作系统win10,不用引入其他的包jdk自带的api就可以 一.java调用ping命令 import jav ...

  2. vue 状态管理 五、Module用法

    系列导航 vue 状态管理 一.状态管理概念和基本结构 vue 状态管理 二.状态管理的基本使用 vue 状态管理 三.Mutations和Getters用法 vue 状态管理 四.Action用法 ...

  3. token原理分析

  4. uni-app editor富文本编辑器

    https://blog.csdn.net/xudejun/article/details/91508189

  5. 基于java+springboot的家教预约网站、家教信息管理系统

    该系统是基于java+springboot开发的家教预约网站.是给师妹开发的实习作品.学习过程中,遇到问题可以在github咨询作者. 演示地址 前台地址: http://jiajiao.gitapp ...

  6. Angular系列教程之MVC模式和MVVM模式

    .markdown-body { line-height: 1.75; font-weight: 400; font-size: 16px; overflow-x: hidden; color: rg ...

  7. 码农的转型之路-IoTBrowser(物联网浏览器)雏形上线

    消失了半个月闭门造轮子去了,最近干了几件大事: 1.工控盒子,win10系统长时间跑物联网服务测试.运行快2周了,稳定性效果还满意,除了windows自动更新重启了一次. 2 .接触了一些新概念MQT ...

  8. [转帖]Linux基础之chkconfig systemd

    https://www.cnblogs.com/barneywill/p/10461279.html   CentOS6服务用chkconfig控制,CentOS7改为systemd控制 1 syst ...

  9. [转帖]Oracle数据库开启NUMA支持

    NUMA简介 NUMA(Non Uniform Memory Access Architecture,非统一内存访问)把一台计算机分成多个节点(node),每个节点内部拥有多个CPU,节点内部使用共有 ...

  10. [转帖]br 备份时排除某个库

    https://tidb.net/book/tidb-monthly/2023/2023-02/usercase/excluded-a-storeroom-during-br-backup 生产环境中 ...