Java中如何实现线程的超时中断

阿里云幸运券

背景
之前在实现熔断降级组件的时候,需要实现接口请求的超时中断。意思是,业务在使用熔断降级功能时,在平台上设置了一个超时时间,如果请求进入熔断器开始计时,接口在超时时间内没有响应,则需要提早中断该请求并返回。

比如正常下游接口的超时时间为800ms,但是因为自身业务的特殊需求,最多只能等200ms,如果指定之内没有数据返回,则返回降级数据。这里处理请求的线程可以看成是tomcat线程池中的一个线程,如果通过线程池返回的Future,可以很轻松的实现超时中断返回,但是当前情况下,并不能拿到Futrue,又不想额外引入一个线程池,所以需要另外一种实现思路。

思路
中断一个线程的思路有哪些?
除了已经废弃的Thread.stop, Thread.suspend, Thread.resume 方法,剩下的貌似只有一种方案了,就是调用当前线程的 interrupt(),但是这个方法的作用并不是中断线程,而是设置一个标识,通知该线程可以被中断了,到底是继续执行,还是中断返回,由线程本身自己决定。

具体来说,当对一个线程调用了 interrupt()之后,如果该线程处于被阻塞状态(比如执行了wait、sleep或join等方法),那么会立即退出阻塞状态,并抛出一个 InterruptedException异常,在代码中catch这个异常进行后续处理。如果线程一直处于运行状态,那么只会把该线程的中断标志设置为 true,仅此而已,所以 interrupt()并不能真正的中断线程,不过在rpc调用的场景中,请求线程一般都处于阻塞状态,等待数据返回,这时 interrupt()方法是可以派上用场的。

那么,要实现指定超时时间内中断请求线程,还有最后一个问题需要解决:什么时候,由谁去执行 interrupt()方法?

必然这个方法只能由其它线程来执行了(自己都阻塞了,执行个鬼),而且是在请求进入熔断器时,并在超时时间之后执行,有点绕,比如超时时间是200ms,那么请求进入熔断器之后,再过200ms,就执行 interrupt(),但是在200ms之内有数据返回,那么就不执行 interrupt()了。

实现
需求已经很明确了,相当于延迟执行一个task,其内部逻辑就是执行请求线程的 interrupt(),当然还有其它的逻辑。

Runnable
task

new

Runnable
()

{

@Override

public

void
run
()

{

try

{

       thread

.
interrupt
();

// 取消定时器任务

       f

.
cancel
();

}

catch

(
Exception
e
)

{

        logger

.
error
(
“Failed while ticking TimerListener”
,
e
);

}

}

};

Doug Lea大神提供的 ScheduledThreadPoolExecutor可以很好的满足这个需求,通过 scheduleAtFixedRate方法可以很方便的实现在延迟指定时间之后执行提交的任务。

ScheduledFuture

<?>
f

executor
.
scheduleAtFixedRate
(

task
,
timeout
,
timeout
,

TimeUnit
.
MILLISECONDS
);

在请求进入熔断器时,顺便提交一个任务到线程池中等待执行,如果接口在超时时间内没有返回,那么该任务会被触发,并执行请求线程的 interrupt方法,这样就实现了请求线程的中断(因为这时请求线程正在被阻塞,等待数据返回),另外需要清空定时任务,不然这个任务会一直执行。

如果接口正常返回了,也要记得清空定时任务,并且在请求退出熔断器的时候,记得恢复请求线程的中断标识,如何恢复?在请求线程中执行下面代码即可。

Thread
.
interrupted
();

// 内部逻辑

public

static

boolean
interrupted
()

{

return
currentThread
().
isInterrupted
(
true
);

}

// 参数为true,可以清除中断标识

private

native

boolean
isInterrupted
(
boolean

ClearInterrupted
);

执行当前线程(即请求线程)的isInterrupted方法。

使用这种方式实现请求的超时中断,在QPS很高的情况下,会有额外的性能损失,因为每次请求都要提交一个任务到线程池中等待执行。

腾讯云代金券

原文链接

https://mp.weixin.qq.com/s?__biz=MzIwMzY1OTU1NQ%3D%3D&mid=2247484184&idx=1&sn=3c40f97e72dc33a429d48e435d93babe&chksm=96cd4354a1baca42ba7797265e4789e05e9a956c415bd2f66aa03cd9db71f4b58b52ced830d9&mpshare=1&scene=23&srcid=0612v8HRIG2g2MmnSfXt7N8F%23rd

服务推荐

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值