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. Spring EL 表达式

    本篇讲述了Spring Expression Language -- 即Spring3中功能丰富强大的表达式语言,简称SpEL. SpEL是类似于OGNL和JSF EL的表达式语言,能够在运行时构建复 ...

  2. vue学习笔记 十三、路由介绍

    系列导航 vue学习笔记 一.环境搭建 vue学习笔记 二.环境搭建+项目创建 vue学习笔记 三.文件和目录结构 vue学习笔记 四.定义组件(组件基本结构) vue学习笔记 五.创建子组件实例 v ...

  3. java基础-IO流-day13

    目录 1. IO的概念 2. 一个一个字符 完成文件的复制 3. 字节流 4. 转换字节流 5. System.in 7.基本数据类型的数据 8. object的处理 1. IO的概念 计算机内存中的 ...

  4. Asp.Net Core造轮之旅:逐步构建自己的开发框架-目录

    本系列适用于已有一定.NET开发基础,学习asp.net core人士. 基础篇 asp.net core之Startup asp.net core之依赖注入 asp.net core之中间件 asp ...

  5. 关于编写C时的调试--VS,VSCODE,DEV-C++

    1.问题 VS最大的问题是不支持scanf函数,后面的调试部分我就没试了,主要用VS来写C++ VSCODE是服了他的C/C++插件,我死活卡在debug中scanf的输入部分,集成终端根本无法输入, ...

  6. [转帖]Linux字符截取命令-cut

    概述 cut是一个选取命令,.一般来说,选取信息通常是针对"行"来进行分析的,并不是整篇信息分析的. 语法 cut [-bn] [file] 1 或 cut [-c] [file] ...

  7. [转帖]使用 TiUP bench 组件压测 TiDB

    https://docs.pingcap.com/zh/tidb/stable/tiup-bench 在测试数据库性能时,经常需要对数据库进行压测,为了满足这一需求,TiUP 集成了 bench 组件 ...

  8. [转帖]公钥基础设施(PKI,Public Key Infrastructure)闲谈

    https://zhuanlan.zhihu.com/p/384436119 背景 在现实空间中,人类的活动范围和接触人的范围有限,人和人最初的信任是建立在小团体或部落内部.随着全球化进展,人类的活动 ...

  9. [转帖]焱融全闪系列科普| 为什么 SSD 需要 NVMe?

    https://xie.infoq.cn/article/7026237b455c7d62f33afc4a9 NVMe 的由来 目前机械硬盘大多数使用 SATA (Serial ATA Advance ...

  10. [转帖]linux性能检测之sar详解

    http://blog.51niux.com/?id=99 sar也是sysstat中的一员. 一.介绍 1.1 简介 sar是一个优秀的一般性能监视工具,它可以输出Linux所完成的几乎所有工作的数 ...