以下内容完全来自于《java核心技术卷Ⅰ 》

    当线程的run方法执行方法体中最后一条语句后,并经由执行return语句返回时,或者出现了在方法中没有捕获的异常时,线程将终止。在Java的早期版本中,还有一个stop方法,其他线程可以调用它终止线程。但是,这个方法现在已经被弃用了。

    有一种可以强制线程终止的方法。然而, interrupt方法可以用来请求终止线程。

    当对一个线程调用interrupt方法时线程的中断状态将被置位。这是每一个线程都具有的boolean标志。每个线程都应该不时地检查这个标志,以判断线程是否被中断。

    要想弄清中断状态是否被置位,首先调用静态的Thread.currentThread方法获得当前线程,然后调用isInterrupted()方法:

    1. while(!Thread.currentThread().isInterrupted() && more work to do){
    2. do more work
    3. }

    但是,如果线程被阻塞,就无法检测中断状态。这是产生InterruptedException异常的地方。当在一个被阻塞的线程(调用sleep或wait)上调用interrupt方法时,阻塞调用将会被InterruptedException异常终点。

    没有任何语言方面的需求要求一个被中断的线程应该终止。中断一个线程不过是引起它的注意。被中断的线程可以决定如何响应中断。某些线程是如此重要以至于应该处理完异常后,继续执行,而不理会中断。但是,更普遍的情况是,线程将简单地将中断作为一个终止的请求。这种线程的run方法具有如下形式:

    1. public void run(){
    2. try{
    3. ...
    4. while(!Thread.currentThread().isInterrupted() && more work to do){
    5. do more work
    6. }
    7. }catch(InterruptedException e){
    8. //thread was interrupted during sleep or wait
    9. }finally{
    10. cleanup , if required
    11. }
    12. //exiting the run method terminates the thread
    13. }

    如果在每次工作迭代之后都调用sleep方法(或者其他的可中断方法), isInterrupted检测既没有必要也没有用处。如果在中断状态被置位时调用sleep方法,它不会休眠。相反,它将清除这一状态(!)并抛出InterruptedException。因此,如果你的循环调用sleep,不会检测中断状态。相反,要如下所示捕获InterruptedException异常:

    1. public void run(){
    2. try{
    3. ...
    4. while(more work to do){
    5. do more work
    6. Thread.sleep(delay);
    7. }
    8. }catch(InterruptedException e){
    9. //thread was interrupted during sleep
    10. }finally{
    11. cleanup, if required
    12. }
    13. //exiting the run method terminateds the thread
    14. }

    注释:有两个非常类似的方法, interrupted和isInterrupted, interrupted方法是一个静态方法,它检测当前的线程是否被中断。而且,调用interrupted方法会清除该线程的中断 “状态。另一方面, isInterrupted方法是一个实例方法,可用来检验是否有线程被中断。调用这个方法不会改变中断状态。

    在很多发布的代码中会发现InterruptedException异常被抑制在很低的层次上,像这样:

    1. void mySubTask{
    2. ...
    3. try{sleep(delay)}
    4. catch(InterruptedExcpetion e{}//DON'T IGNORE!
    5. ...
    6. }

    不要这样做!如果不认为在catch子句中做这一处理有什么好处的话,仍然有两种合理的选择:

    1. 在catch子句中调用Thread.currentThread().interrupt来设置中断状态。于是,调用者可以对其进行检测。
    1. void mySubTask{
    2. ...
    3. try{sleep(delay)}
    4. catch(InterruptedExcpetion e{
    5. Thread.currentThread.interrupt();
    6. }
    7. ...
    8. }
    1. 或者,更好的选择是,用throws InterruptedException将异常抛出,留给调用者。

    void mySubTask{

    try{sleep(delay)}
    catch(InterruptedExcpetion e{}//DON’T IGNORE!

    }

    • void interrupt()
      向线程发送中断请求,线程的中断状态将被设置为true,如果目前该线程被一个sleep调用阻塞,那么, InterruptedException异常被抛出。
    • static boolean interrupted()
      测试当前线程(即正在执行这一命令的线程)是否被中断。注意,这是一个静态方法。”这一调用会产生副作用-它将当前线程的中断状态重置为false。
    • boolean isInterrupted()
      测试线程是否被终止。不像静态的中断方法,这一调用不改变线程的中断状态。
    • static Thread currentThread()
      返回代表当前执行线程的Thread对象