引子

这个题也算面试常见题了,如果你回答 Thread.stop(),那么面试差不多要结束了......这里先给出答案

  1. 给线程发送中断
  2. 在自己的业务代码里,做到以下 2 件事
  • 捕获 InterruptedException 异常,退出线程的执行
  • 调用 isInterrupted 方法,若结果为 true,退出线程的执行

接下来详细说明,并延伸谈谈为什么要停止一个执行中的线程

阻塞

我们先要明白什么是阻塞,答案有很多,这里给出一个最简单的说法

凡是抛出 InterruptedException 异常的操作,都会导致阻塞,例如

Thread.sleep();Object.wait();Lock.lockInterruptibly();BlockingQueue.take();Future.get();

那么,问题来了:网络通信时,等待服务端返回数据是不是阻塞?

答案是:NO,因为这时抛出的异常是 IOException 而不是 InterruptedException

阻塞含义:抛出 InterruptedException 异常的操作

为什么我们要搞明白什么是阻塞呢?原因是 Thread.interrupt 就是我们退出阻塞的唯一方法(除非你的阻塞操作自带了超时检测)

如果一个执行中的线程正处于阻塞状态,要停止它,有且仅有一个办法:来一个 interrupt,然后捕获 InterruptedException 异常并退出

这也意味着要中断网络 IO,并没有那么 easy

非阻塞

如果是非阻塞的状态下,怎么停止线程呢?

这里没有银弹,首先还是来一个 interrupt,然后重要的是,在你的业务代码的任意节点(一般是关键节点),检查中断状态,例如下面的代码

if (Thread.currentThread().isInterrupted()) {    return;}

比如,我们刚才说过,网络通信时在等待服务器返回数据,如果这时你想停止,那么只能是在服务器返回数据后,检查中断状态,然后决定是处理服务器返回的数据,还是直接退出;当然,基于中断发送的时机,如果这时你还没有和远程服务器连接,那就可以直接退出了

或者,你的业务代码正在执行循环,可以在每次循环的开始处进行检查

如何发送中断

要停止一个执行中的线程,第一个步骤就是发送一个中断给线程,有以下几种方式

  • 如果你能拿到 thread 对象,调用其 interrupt 方法
  • 如果你拿到的是线程池返回的 Future,调用其 cancel 方法,请自行研究下该方法传递的参数
  • 当然,还有个玉石俱焚的方法:直接把线程池给 shutdown......请自行研究下 ExecutorService 的 shutdown 和 shutdownNow

为什么要停止执行中的线程

  • 首先,线程是一种资源,如果是从线程池里获取的,那么还是一种共享资源,提前停止执行,可以更早的归还到池里,增加整个线程池的吞吐率
  • 其次,需要被停止的操作,多半是无意义的操作,除了耗费 cpu 资源,占用更多的内存以外,毫无意义

举例说明

在广告实时竞价时,需要预估广告的转化率 CTR,这个 CTR 一般都是由一个专门的预估服务提供的,在我们的 dsp 实现中,每次竞价会向 2 个不同的 CTR 预估服务发起调用,这样某一个服务超时,可以采用另一个服务的预估数据

2 个预估服务是有优先级的,如果优先级更高的服务已经返回结果,那么另一个服务的返回结果就没有意义了,如果该服务尚未返回,我们就应该停止它

一次广告竞价,参与的广告往往成千上万,每个广告都要给出一个预估值,预估服务每次返回的数据量是极大的,能到几十上百 K,然后我们还要对该数据进行解析:比如先把字节流转为字符串,再用 json 格式解析字符串,得到一个有数千元素的列表

如果我们在预估服务返回数据,但还没有执行字节流转字符串时,就发送中断,那么后续生成字符串并解析的操作就无需进行,这样很明显就节约了内存




java线程 防止重复执行 java线程如何停止_java线程 防止重复执行