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. 机器学习-决策树系列-决策树-剪枝-CART算法-27

    目录 1. 剪枝 2. CCP-代价复杂度剪枝(CART) 4. α值的确定 1. 剪枝 将子树还原成一个叶子节点: 是解决过拟合的一个有效方法.当树训练得过于茂盛的时候会出现在测试集上的效果比训练集 ...

  2. idea相关配置及插件安装

    对idea相关的配置及好用的插件进行总结下. 一.idea 破解码及配置:https://www.jb51.net/softs/672190.html 二.idea插件: 1.findBugs-ide ...

  3. Visual Studio实用的搜索、查找、替换技巧

    前言 对于.NET开发者而言Visual Studio是我们日常工作中比较常用的开发工具,掌握一些Visual Studio实用的搜索.查找.替换技巧可以帮助我们大大提高工作效率从而避免996. Vi ...

  4. Redis之入门概括与指令

    Redis特点(AP模型,优先保证可用,不会管数据丢失): 快的原因: 基于内存操作,操作不需要跟磁盘交互 k-v结构,类似与hashMap,所以查询速度非常快,接近O(1). 底层数据结构是有如:跳 ...

  5. Go-错误栈信息

    Go中错误栈信息 .\main.go:22:2: number1 declared but not used .\main.go 错误发生的文件 22:2 文件的22行第2列 number1 decl ...

  6. [转帖]浏览器HTTP请求并发数和TCP连接的关系

    https://cloud.tencent.com/developer/article/1518678 面试题目(头条): 网页中的图片资源为什么分放在不同的域名下? 浏览器与服务器建立一个TCP连接 ...

  7. [转帖]rsync原理

    简介: Rsync(remote synchronize)是一个远程数据同步工具,可通过LAN/WAN快速同步多台主机间的文件.Rsync使用所谓的"Rsync算法"来使本地和远 ...

  8. Harbor修改默认网段以及设置开机启动的方法

    Harbor修改默认网段以及设置开机启动的方法 背景 docker 默认的网段是 172.16.xx.xx 网段. harbor进行设置时会自动加1 设置为 172.17.xx.xx 有时候这个网段是 ...

  9. [转帖]Shell脚本数组(实现冒泡排序,直接选择排序,反转排序)

    目录 数组 数组定义方法 数组包括的数据类型 获取数组长度 读取某下标赋值 数组遍历 数组切片 数组替换 删除数组 追加数组中的元素 从函数返回数组 加法传参运算 乘法传参运算 数组排序算法 冒泡排序 ...

  10. [转帖]jmeter_采样器sampler简介

    1.取样器介绍 取样器是用来模拟用户操作的,向服务器发送请求以及接收服务器的响应数据. 取样器是在线程组内部的元件,也就是说取样器只能在线程组中添加. 取样器(Sampler)是性能测试中向服务器发送 ...