JDK8优雅地实现异步线程超时熔断,返回默认值或者抛出异常

在JDK8中进行多线程调用时,如果想指定一个超时时间,若子线程执行超时则直接熔断处理,该怎么优雅地实现呢?

从Java 8开始引入了CompletableFuture,它针对Future做了改进,可以传入回调对象,当异步任务完成或者发生异常时,自动调用回调对象的回调方法。

但是,在JDK8中没有提供子线程超时处理机制。试着看了下JDK9的代码,果然有了相应的实现,具体可参考JDK9的CompletableFuture类。下面是我将其中的部分代码提取出来实现了一个增强类,可以在JDK8中正常运行。

import java.util.Arrays;
import java.util.concurrent.*;
import java.util.function.BiConsumer;

/**
 * CompletableFuture功能增强类
 * <p>对jdk8中的CompletableFuture进行功能增强</p>
 * <p>参考jdk9</p>
 *
 * @author wangyu
 * @date 2021/9/24 10:51
 */
public class CompletableFutureHelper {
    /**
     * 单例延时调度程序,仅用于启动和取消任务
     */
    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 {
            @Override
            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 CompletableFutureHelper.Delayer.DaemonThreadFactory())).
                    setRemoveOnCancelPolicy(true);
        }
    }

    /**
     * 超时时以抛异常的形式结束任务
     */
    static final class Timeout implements Runnable {
        final CompletableFuture<?> f;

        Timeout(CompletableFuture<?> f) {
            this.f = f;
        }

        @Override
        public void run() {
            if (f != null && !f.isDone()) {
                f.completeExceptionally(new TimeoutException());
            }
        }
    }

    /**
     * 在超时时完成
     */
    static final class DelayedCompleter<U> implements Runnable {
        final CompletableFuture<U> f;
        final U u;

        DelayedCompleter(CompletableFuture<U> f, U u) {
            this.f = f;
            this.u = u;
        }

        @Override
        public void run() {
            if (f != null) {
                f.complete(u);
            }
        }
    }

    /**
     * 取消不需要的超时任务
     */
    static final class Canceller implements BiConsumer<Object, Throwable> {
        final Future<?> f;

        Canceller(Future<?> f) {
            this.f = f;
        }

        @Override
        public void accept(Object ignore, Throwable ex) {
            if (ex == null && f != null && !f.isDone()) {
                f.cancel(false);
            }
        }
    }

    /**
     * 若执行超时则返回默认值,否则返回计算出来的结果值
     *
     * @param future  源future
     * @param value   超时返回的默认值
     * @param timeout 超时时间
     * @param unit    超时时间单位
     * @param <T>
     * @return
     */
    public static <T> CompletableFuture<T> completeOnTimeout(CompletableFuture<T> future, T value, long timeout, TimeUnit unit) {
        if (future == null || unit == null) {
            throw new NullPointerException();
        }
        if (null == future.getNow(null)) {
            future.whenComplete(new Canceller(Delayer.delay(new DelayedCompleter<T>(future, value), timeout, unit)));
        }
        return future;
    }

    /**
     * 若执行超时则抛出异常,否则返回计算出来的结果值
     *
     * @param future  源future
     * @param timeout 超时时间
     * @param unit    超时时间单位
     * @param <T>
     * @return
     */
    public static <T> CompletableFuture<T> orTimeout(CompletableFuture<T> future, long timeout, TimeUnit unit) {
        if (future == null || unit == null) {
            throw new NullPointerException();
        }
        if (null == future.getNow(null)) {
            future.whenComplete(new Canceller(Delayer.delay(new Timeout(future), timeout, unit)));
        }
        return future;
    }

    /**
     * 包装所有的任务,加上超时处理,超时的任务默认返回null
     *
     * @param timeout 超时时间
     * @param unit    超时时间单位
     * @param cfs     任务
     * @return CompletableFuture<Void>
     * @see CompletableFuture#allOf(CompletableFuture[])
     */
    public static CompletableFuture<Void> allOfWithCompleteOnTimeout(long timeout, TimeUnit unit, CompletableFuture<?>... cfs) {
        Arrays.stream(cfs).forEach(cf -> cf = completeOnTimeout(cf, null, timeout, unit));
        return CompletableFuture.allOf(cfs);
    }

}

测试代码如下:

public static void main(String[] args) throws ExecutionException, InterruptedException, IOException {
        long start = System.currentTimeMillis();

        CompletableFuture future1 = CompletableFuture.supplyAsync(()-> {
            sleep(3L);
            System.out.printf("future1 [%s] end\n", Thread.currentThread().getName());
            return "[data] future1";
        });
        CompletableFuture future2 = CompletableFuture.supplyAsync(()-> {
            sleep(8L);
            System.out.printf("future2 [%s] end\n", Thread.currentThread().getName());
            return "[data] future2";
        });
//        CompletableFuture.allOf(future1, future2).join();

        System.out.println("main get start...");

        //场景1:包装任务,加上超时处理,超时的任务默认返回null
//        future1 = CompletableFutureHelper.completeOnTimeout(future1, null, 4, TimeUnit.SECONDS);
//        future2 = CompletableFutureHelper.completeOnTimeout(future2, null, 4, TimeUnit.SECONDS);
//        CompletableFuture.allOf(future1, future2).join();

        //场景2:包装任务,加上超时处理,任一任务超时则抛错TimeoutException
//        future1 = CompletableFutureHelper.orTimeout(future1, 4, TimeUnit.SECONDS);
//        future2 = CompletableFutureHelper.orTimeout(future2, 4, TimeUnit.SECONDS);
//        CompletableFuture.allOf(future1, future2).join();

        //此一行代码等价于场景一3行代码,使用更方便;不过场景一中可以分别指定不同的默认返回值,更加灵活
        CompletableFutureHelper.allOfWithCompleteOnTimeout(4, TimeUnit.SECONDS, future1, future2).join();
        //超时时间设置成7995毫秒,执行时间为8s的线程不一定会超时(本次测试精度大概在10ms左右,7990时,执行8s的任务大概率当做超时处理)
//      CompletableFutureHelper.allOfWithCompleteOnTimeout(7995, TimeUnit.MILLISECONDS, future1, future2).join();

        System.out.println("[get future1] " + future1.get());
        System.out.println("[get future2] " + future2.get());

        System.out.println("main get end...");
        System.out.printf("main [%s] end cost %d ms\n", Thread.currentThread().getName(), System.currentTimeMillis()-start);
        System.in.read();
    }
    
    private static void sleep(long seconds) {
        try {
            TimeUnit.SECONDS.sleep(seconds);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

测试代码打印内容如下

main get start...
future1 [ForkJoinPool.commonPool-worker-1] end
[get future1] [data] future1
[get future2] null
main get end...
main [main] end cost 4057 ms
future2 [ForkJoinPool.commonPool-worker-2] end
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,271评论 5 466
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,725评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,252评论 0 328
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,634评论 1 270
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,549评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 47,985评论 1 275
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,471评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,128评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,257评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,233评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,235评论 1 328
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,940评论 3 316
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,528评论 3 302
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,623评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,858评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,245评论 2 344
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,790评论 2 339

推荐阅读更多精彩内容