同步模式之保护性暂停模式

  1. Guarded Suspension:一个线程需要等待另一个线程的执行结果

  2. 理解

  • 一个线程需要将结果传递给另一个线程,将这两个线程关联到到同一个 GuardedObject
  • 如果需要源源不断地传递结果,需要使用消息队列(生产者-消费者模型)
  • JDK 中,join 的实现、Future 的实现用的都是保护性暂停模式
  • 因为要等待,所以算作同步模式
  1. 实现
  • 创建两个线程和一个守护对象,线程 1 wait 等待下载结果 response,线程 2 执行下载,下载完成将结果赋值给 GuardedObject 的 response 属性,然后 notify 唤醒线程 1,线程 1 被唤醒,继续执行,两个线程通过 GuardedObject 传递要用的结果
  1. 优点
  • join 需要等一个线程运行结束了才能得到该线程的结果
  • join 等待结果的变量只能是全局的,GuardedObject 可以是局部的
  1. 扩展 1
  • 超时退出:wait(timeout) 只等待一段时间,如果超时了还没有获取到结果就直接退出循环

(resposne 是下载完成之后赋值的,response 不为空之后就可以退出循环,返回 res)

**6. join 的原理:**保护性暂停是等待一个线程的结果,join 是等待一个线程的结束

  • 就是应用了保护性暂停模式,如果**线程还存活(对应上面的下载结果 response == null)**且未超时就一直等待,如果超时了(还没有得到结果)就退出循环

    复制代码
      /**
       * Waits at most {@code millis} milliseconds for this thread to
       * die. A timeout of {@code 0} means to wait forever.
       *
       * <p> This implementation uses a loop of {@code this.wait} calls
       * conditioned on {@code this.isAlive}. As a thread terminates the
       * {@code this.notifyAll} method is invoked. It is recommended that
       * applications not use {@code wait}, {@code notify}, or
       * {@code notifyAll} on {@code Thread} instances.
       *
       * @param  millis
       *         the time to wait in milliseconds
       *
       * @throws  IllegalArgumentException
       *          if the value of {@code millis} is negative
       *
       * @throws  InterruptedException
       *          if any thread has interrupted the current thread. The
       *          <i>interrupted status</i> of the current thread is
       *          cleared when this exception is thrown.
       */
      public final synchronized void join(long millis)
      throws InterruptedException {
          long base = System.currentTimeMillis();
          long now = 0;
    
          if (millis < 0) {
              throw new IllegalArgumentException("timeout value is negative");
          }
    
          if (millis == 0) {
              while (isAlive()) {
                  wait(0);
              }
          } else {
              while (isAlive()) {
                  long delay = millis - now;
                  if (delay <= 0) {
                      break;
                  }
                  wait(delay);
                  now = System.currentTimeMillis() - base;
              }
          }
      }
  1. 扩展2:生产者消费者节耦
相关推荐
Tiger Z2 分钟前
R 语言科研绘图第 41 期 --- 桑基图-基础
开发语言·r语言·贴图
打死不学Java代码3 分钟前
PaginationInnerInterceptor使用(Mybatis-plus分页)
android·java·mybatis
南客先生5 分钟前
海量聊天消息处理:ShardingJDBC分库分表、ClickHouse冷热数据分离、ES复合查询方案、Flink实时计算与SpringCloud集成
java·clickhouse·elasticsearch·flink·springcloud·shardingjdbc
w236173460112 分钟前
Tomcat:从零理解Java Web应用的“心脏”
java·前端·tomcat
chuxinweihui15 分钟前
数据结构——二叉树,堆
c语言·开发语言·数据结构·学习·算法·链表
yuren_xia17 分钟前
示例:Spring JDBC编程式事务
java·后端·spring
陈大大陈32 分钟前
基于 C++ 的用户认证系统开发:从注册登录到Redis 缓存优化
java·linux·开发语言·数据结构·c++·算法·缓存
纪元A梦34 分钟前
华为OD机试真题——通过软盘拷贝文件(2025A卷:200分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
java·javascript·c++·python·华为od·go·华为od机试题
看到我,请让我去学习41 分钟前
C语言基础(day0424)
c语言·开发语言·数据结构
studyer_domi43 分钟前
Matlab 复合模糊PID
开发语言·matlab