【异步编程实战】如何实现超时功能(以CompletableFuture为例)

embedded/2025/1/18 7:37:38/

【异步编程实战】如何实现超时功能(以CompletableFuture为例)

【异步编程实战】如何实现超时功能(以CompletableFuture为例)

  • 由于网络波动或者连接节点下线等种种问题,对于大多数网络异步任务的执行常常会进行超时限制,在异步开发中可以看成是一个常见的问题。本文主要讨论实现超时功能的基本思路以及CompletableFuture(之后简称CF)是如何通过代码实现超时功能的。

基本思路

  • 两个任务,两个线程:原有任务,超时任务
  • 原有的任务正常执行,写入正常结果,原有任务执行成功取消超时任务
  • 超时时取消原有任务,写入结果为超时异常或者默认值
  • 静态条件下保证结果写入的原子性和只写一次

CompletableFuture 的实现

基本实现流程

  • // JDK9新增的超时方法
    public CompletableFuture<T> orTimeout(long timeout, TimeUnit unit) {if (unit == null)throw new NullPointerException();if (result == null)whenComplete(new Canceller(Delayer.delay(new Timeout(this),timeout, unit)));return this;
    }// CF的内部类static final class Timeout implements Runnable {final CompletableFuture<?> f;Timeout(CompletableFuture<?> f) { this.f = f; }public void run() {if (f != null && !f.isDone())f.completeExceptionally(new TimeoutException());}}
    
  • 分析代码得知,whenComplete方法添加了正常结束的回调,取消超时任务。

  • 超时任务通过Delayer.delay创建,超时时执行Timeout::run方法,即写入结果为TimeoutException。

  • 下面来看下Dalayer的具体实现:

  • /*** Singleton delay scheduler, used only for starting and* cancelling tasks.*/
    static final class Delayer {static ScheduledFuture<?> delay(Runnable command, long delay,TimeUnit unit) {return delayer.schedule(command, delay, unit);}static final class DaemonThreadFactory implements ThreadFactory {public Thread newThread(Runnable r) {Thread t = new Thread(r);// 守护线程,当主线程关闭时,自身也关闭t.setDaemon(true);t.setName("CompletableFutureDelayScheduler");return t;}}static final ScheduledThreadPoolExecutor delayer;static {(delayer = new ScheduledThreadPoolExecutor(1, new DaemonThreadFactory())).setRemoveOnCancelPolicy(true);}
    }
    
  • Delayer是一个单例对象,专门用于执行延迟任务,减少了内存占用。ScheduledThreadPoolExecutor 的配置为单线程,设置了removeOnCancelPolicy,表示取消延迟任务时,任务从延迟队列删除。这里的延迟队列为默认的执行器实现:

  • public ScheduledThreadPoolExecutor(int corePoolSize,ThreadFactory threadFactory) {super(corePoolSize, Integer.MAX_VALUE,DEFAULT_KEEPALIVE_MILLIS, MILLISECONDS,new DelayedWorkQueue(), threadFactory);
    }
    
  • ScheduledThreadPoolExecutor 底层使用延迟队列DelayedWorkQueue,延迟队列底层依赖于索引优先队列,删除操作的时间复杂度为o(logn)。

  • 下面来看下Canceller的具体实现:

  • static final class Canceller implements BiConsumer<Object, Throwable> {final Future<?> f;Canceller(Future<?> f) { this.f = f; }public void accept(Object ignore, Throwable ex) {if (f != null && !f.isDone())f.cancel(false);}
    }
    
  • canceller实际上是一个回调函数,原有任务完成后触发,会取消相关超时任务。

静态条件分析

  • 下面是写入CF的实现代码片段:

  • 				// 超时结束        if (f != null && !f.isDone())f.completeExceptionally(new TimeoutException());// 取消任务if (f != null && !f.isDone())f.cancel(false);// CF 原有任务的写入不由orTimeout方法控制,以下为一个示例Thread.sleep(1000);f.complete(u);
    
  • 对于CF的检查实际上不能保证原子性,因为这种检查-再计算的模式需要同步块的保护,而CF底层并没有这种实现。所以,if语句检查任务未完成,之后执行代码时,任务可能已经完成了。不过这种检查也有一定的好处,因为CF保证了结果写入后,isDone方法必然为true,从而避免执行不必要的代码。

  • completeExceptionally 方法和 complete 方法可能同时执行,CF 通过CAS操作保证了结果写入的原子性。

  • // 异常结果实现
    final boolean internalComplete(Object r) { // CAS from null to rreturn RESULT.compareAndSet(this, null, r);
    }
    // 正常结果实现
    final boolean completeValue(T t) {return RESULT.compareAndSet(this, null, (t == null) ? NIL : t);
    }public boolean isDone() {return result != null;
    }
    

内存泄露bug

  • 在 JDK21之前的CF实现中,存在内存泄露的bug,作为bug,后续发布的 JDK 子版本可能会修复这个问题。

  • 这个bug在如下代码中:

  • // 取消任务,JDK21之前的实现会检查异常结果
    if (ex == null && f != null && !f.isDone())f.cancel(false);
    
  • 当正常任务异常结束时,不会取消延迟队列中的任务,最终会导致内存泄露。若项目中存在多个长时间超时CF任务,内存泄露的情况会更明显。

  • public class LeakDemo {public static void main(String[] args) {while (true) {new CompletableFuture<>().orTimeout(1, TimeUnit.HOURS).completeExceptionally(new Exception());}}
    }
    
  • 执行以上代码会报OOM错误,你可以在自己的编程环境中进行测试。


http://www.ppmy.cn/embedded/115480.html

相关文章

js进阶——什么是提升

JavaScript 中的提升&#xff08;Hoisting&#xff09;是一个重要的概念&#xff0c;它指的是 JavaScript 引擎在代码执行之前&#xff0c;会将变量和函数的声明提升到它们所在作用域的顶部。这意味着&#xff0c;即使变量或函数在代码中后面声明&#xff0c;它们的引用可以在声…

Hive之任务优化

Hive 是一个基于 Hadoop 的数据仓库工具&#xff0c;提供了 SQL-like 的查询语言来分析存储在 HDFS&#xff08;Hadoop Distributed File System&#xff09;上的大规模数据集。为了提高查询性能&#xff0c;Hive 提供了多种优化方法&#xff0c;涵盖不同层次的改进&#xff0c…

软件测试技术之 GPU 单元测试是什么!

1 背景 测试是开发的一个非常重要的方面&#xff0c;可以在很大程度上决定一个应用程序的命运。良好的测试可以在早期捕获导致应用程序崩溃的问题&#xff0c;但较差的测试往往总是导致故障和停机。 单元测试用于测试各个代码组件&#xff0c;并确保代码按照预期的方式工作。单…

证书学习(五)Java实现RSA、SM2证书颁发

目录 一、知识回顾1.1 X.509 证书1.2 X509Certificate 类二、代码实现2.1 Maven 依赖2.2 RSA 证书颁发1)PfxGenerateUtil 证书文件生成工具类2)CertDTO 证书中间类3)RSACertGenerateTest RSA证书生成测试类4)执行结果2.3 SM2 证书颁发1)SM2Utils 国密SM2算法工具类2)SM2C…

k8s的一些命令

kubectl get nodes &#xff1a;查看节点的状态 查看Pod的状态&#xff1a; kubectl get pod --all -namespacesPending,ContainerCreating,ImagePullBackOff都表明Pod没有就绪&#xff0c;Running才是就绪状态 查看Pod的具体情况&#xff1a; kubectl describe pod podnamek…

PDF转图片的思路思考

记录时间:2022年9月1日 PDF转图片库的使用和扩展 python有几个开源的免费的处理Pdf的库&#xff0c;甚至有的已经有很完善的功能了。我发挥一下自己的所学&#xff0c;看看能不能把它变为可用的一程序。 首先是了解PDF处理库PyMupdf&#xff0c;这个库得到路径之后普就可以对…

Vulkan 学习(8)---- vkImageView 创建

目录 OverView创建方法关键结构参考代码 OverView Vulkan 的图像视图(VkImageView) 用于描述如何访问 VkImage 对象以及访问图像的哪一部分, 图像视图定义了图像格式和访问方式&#xff0c;允许渲染管线和图像进行交互&#xff0c;无论是作为纹理&#xff0c;颜色附件或者深度…

Elastic 的 OpenTelemetry PHP 发行版简介

作者&#xff1a;Pawel Filipczak 宣布 OpenTelemetry PHP 的 Elastic 发行版的第一个 alpha 版本。在本篇博文中了解使用 OpenTelemetry 来检测 PHP 应用程序是多么简单。 我们很高兴推出 OpenTelemetry PHP 的 Elastic Distribution 的第一个 alpha 版本。在这篇文章中&…