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. 【调试】netconsole的使用

    开发环境 客户端 开发板:FireFly-RK3399 Linux 4.4 IP:192.168.137.110 服务端 VMware Workstation Pro16,ubuntu 18.04 I ...

  2. FrameWork使用TraeFik连接Grpc的坑

    背景介绍:因为公司最近使用TraeFik来代替nginx做代理服务器.导致一些老项目访问Grpc的时候直接Status(StatusCode=Unavailable, Detail="fai ...

  3. plsql打开报错:Control 'dxDockBrowserPanel' has no parent window问题解决

    一.现象: 使用plsql登陆oracle数据库时,登陆信息没有报错,但是最后一步报错,重启电脑依然没有解决 一直报:" Control 'dxDockBrowserPanel' has n ...

  4. Ubuntu解决Github无法访问的问题

    技术背景 由于IP设置的问题,有时候会出现Github无法访问的问题,经过一番的资料检索之后,发现如下的方案可以成功解决在Ubuntu下无法正常访问Github的问题(有时候可以打开,有时候又不行). ...

  5. Icoding 链表 删除范围内结点

    1.题目: 已知线性表中的元素(整数)以值递增有序排列,并以单链表作存储结构.试写一高效算法,删除表中所有大于mink且小于maxk的元素(若表中存在这样的元素),分析你的算法的时间复杂度. 链表结点 ...

  6. 在Windows 版的Chrome中切换标签页

    按住Ctrl+tab键之后使用page up/ page down键左右移动!

  7. CSS - 设置自动等比例缩放

    img {     width: 100vw;     height: 100vh;     object-fit: cover;  }

  8. Go-值传递&引用传递

    值类型和引用类型 值类型关注其值 引用类型关注其指针 值类型和引用类型区别在于传递值的时候拷贝的对象不同,值传递拷贝是变量的值,引用传递拷贝的是变量的指针 拷贝 -- 传递值 赋值 函数调用 初始化 ...

  9. [转帖]故障分析 | 让top命令直接显示Java线程名?-- 解析OpenJDK的一个bug修复

    https://zhuanlan.zhihu.com/p/413136873 作者:阎虎青DBLE 开源项目负责人,负责分布式数据库中间件研发工作:持续专注于数据库方面的技术,始终在一线从事开发:对数 ...

  10. [转帖]一图胜千言 -- SQL Server 基准测试

    https://blog.51cto.com/ultrasql/2130487 文章标签基准测试文章分类SQL Server数据库阅读数1116