对Thread.interrupt()方法很详细的介绍

在JDK1.0中,可以用stop方法来终止,但是现在这种方法已经被禁用了,改用interrupt方法。

Thread.interrupt()方法不会中断一个正在运行的线程。它的作用是,在线程受到阻塞时抛出一个中断信号,这样线程就得以退出阻塞的状态。更确切的说,如果线程被Object.wait, Thread.join和Thread.sleep三种方法之一阻塞,那么,它将接收到一个中断异常(InterruptedException),从而提早地终结被阻塞状态。

interrupt方法并不是强制终止线程,它只能设置线程的interrupted状态,而在线程中一般使用一下方式:
while (!Thread.currentThread().isInterrupted() && more work to do)

{...}

而被block的线程(sleep() or join())在被调用interrupt时会产生InterruptException,此时是否终止线程由本线程自己决定。程序的一般形式是:
public void run()
{
try
{
. . .
while (!Thread.currentThread().isInterrupted() && more work to do)
{
do more work
}
}
catch(InterruptedException e)
{
// thread was interrupted during sleep or wait
}
finally
{
cleanup, if required
}
// exiting the run method terminates the thread
}

Thread.sleep方法也会产生InterruptedException,因此,如果每次在做完一些工作后调用了sleep方法,那么就不用检查isInterrupted,而是直接捕捉InterruptedException。

---------------------------------------------------------------------------------------

假如我们有一个任务如下,交给一个Java线程来执行,如何才能保证调用interrupt()来中断它呢?

class ATask implements Runnable{  
  
    private double d = 0.0;  
      
    public void run() {  
        //死循环执行打印"I am running!" 和做消耗时间的浮点计算  
        while (true) {  
            System.out.println("I am running!");  
              
            for (int i = 0; i < 900000; i++) {  
                d =  d + (Math.PI + Math.E) / d;  
            }  
            //给线程调度器可以切换到其它进程的信号  
            Thread.yield();  
        }  
    }  
}  
  
public class InterruptTaskTest {  
      
    public static void main(String[] args) throws Exception{  
        //将任务交给一个线程执行  
        Thread t = new Thread(new ATask());  
        t.start();  
          
        //运行一断时间中断线程  
        Thread.sleep(100);  
        System.out.println("****************************");  
        System.out.println("Interrupted Thread!");  
        System.out.println("****************************");  
        t.interrupt();  
    }  
}   


运行这个程序,我们发现调用interrupt()后,程序仍在运行,如果不强制结束,程序将一直运行下去,如下所示: 
......  
I am running!  
I am running!  
I am running!  
I am running!  
****************************  
Interrupted Thread!  
****************************  
I am running!  
I am running!  
I am running!  
I am running!  
I am running!  
....  

虽然中断发生了,但线程仍然在进行,离开线程有两种常用的方法: 
抛出InterruptedException和用Thread.interrupted()检查是否发生中断,下面分别看一下这两种方法: 
1.在阻塞操作时如Thread.sleep()时被中断会抛出InterruptedException(注意,进行不能中断的IO操作而阻塞和要获得对象的锁调用对象的synchronized方法而阻塞时不会抛出InterruptedException) 
Java代码 
class ATask implements Runnable{  
  
    private double d = 0.0;  
      
    public void run() {  
        //死循环执行打印"I am running!" 和做消耗时间的浮点计算  
        try {  
            while (true) {  
                System.out.println("I am running!");  
                  
                for (int i = 0; i < 900000; i++) {  
                    d =  d + (Math.PI + Math.E) / d;  
                }  
                //休眠一断时间,中断时会抛出InterruptedException  
                Thread.sleep(50);  
            }  
        } catch (InterruptedException e) {  
            System.out.println("ATask.run() interrupted!");  
        }  
    }  
}  

程序运行结果如下: 
Java代码 
I am running!  
I am running!  
****************************  
Interrupted Thread!  
****************************  
ATask.run() interrupted!  

可以看到中断任务时让任务抛出InterruptedException来离开任务. 

2.Thread.interrupted()检查是否发生中断.Thread.interrupted()能告诉你线程是否发生中断,并将清除中断状态标记,所以程序不会两次通知你线程发生了中断. 
Java代码 
class ATask implements Runnable{  
  
    private double d = 0.0;  
      
    public void run() {  
          
        //检查程序是否发生中断  
        while (!Thread.interrupted()) {  
            System.out.println("I am running!");  
  
            for (int i = 0; i < 900000; i++) {  
                d = d + (Math.PI + Math.E) / d;  
            }  
        }  
  
        System.out.println("ATask.run() interrupted!");  
    }  
}  

程序运行结果如下: 
Java代码 
I am running!  
I am running!  
I am running!  
I am running!  
I am running!  
I am running!  
I am running!  
****************************  
Interrupted Thread!  
****************************  
ATask.run() interrupted!  


我们可结合使用两种方法来达到可以通过interrupt()中断线程.请看下面例子: 
Java代码 
class ATask implements Runnable{  
  
    private double d = 0.0;  
      
    public void run() {  
          
        try {  
        //检查程序是否发生中断  
        while (!Thread.interrupted()) {  
            System.out.println("I am running!");  
            //point1 before sleep  
            Thread.sleep(20);  
            //point2 after sleep  
            System.out.println("Calculating");  
            for (int i = 0; i < 900000; i++) {  
                d = d + (Math.PI + Math.E) / d;  
            }  
        }  
          
        } catch (InterruptedException e) {  
            System.out.println("Exiting by Exception");  
        }  
          
        System.out.println("ATask.run() interrupted!");  
    }  
}  

在point1之前处point2之后发生中断会产生两种不同的结果,可以通过修改InterruptTaskTest main()里的Thread.sleep()的时间来达到在point1之前产生中断或在point2之后产生中断. 
如果在point1之前发生中断,程序会在调用Thread.sleep()时抛出InterruptedException从而结束线程.这和在Thread.sleep()时被中断是一样的效果.程序运行结果可能如下:
Java代码 
I am running!  
Calculating  
I am running!  
Calculating  
I am running!  
Calculating  
I am running!  
****************************  
Interrupted Thread!  
****************************  
Exiting by Exception  
ATask.run() interrupted!  

如果在point2之后发生中断,线程会继续执行到下一次while判断中断状态时.程序运行结果可能如下: 
Java代码 
I am running!  
Calculating  
I am running!  
Calculating  
I am running!  
Calculating  
****************************  
Interrupted Thread!  
****************************  
ATask.run() interrupted!  

### Java 中 `Thread.currentThread().interrupt()` 的使用说明 `Thread.currentThread().interrupt()` 是一种用于设置当前线程中断状态的方法。它并不会立即终止线程,而是通过改变线程的中断标志位来通知该线程应该停止其工作[^3]。 #### 1. **基本概念** 当调用 `Thread.currentThread().interrupt()` 时,如果当前线程处于阻塞状态(如等待锁、休眠或 I/O 操作),则会抛出 `InterruptedException` 并清除中断标志;如果线程未处于阻塞状态,则仅设置其中断标志而不触发任何异常。 以下是具体的行为总结: - 如果线程正在执行某些可能抛出 `InterruptedException` 的操作(例如 `Thread.sleep(long millis)` 或 `Object.wait()`),那么这些方法会在检测到中断后抛出 `InterruptedException`。 - 对于非阻塞的操作,调用 `interrupt()` 只会更改线程的中断状态,不会直接影响线程行为。 #### 2. **代码示例** 以下是一个简单的例子展示了如何使用 `Thread.currentThread().interrupt()`: ```java public class InterruptExample { public static void main(String[] args) throws InterruptedException { Thread worker = new Thread(() -> { try { System.out.println("Worker thread started."); // Simulate some work that can be interrupted. while (!Thread.currentThread().isInterrupted()) { System.out.println("Working..."); Thread.sleep(1000); } System.out.println("Worker thread received interrupt signal and will terminate gracefully."); } catch (InterruptedException e) { System.out.println("Worker thread was interrupted during sleep."); Thread.currentThread().interrupt(); // Restore the interrupt status. } }); worker.start(); // Let the worker do its job for a few seconds before we ask it to stop. Thread.sleep(3000); System.out.println("Main thread requests worker thread to stop."); // Requesting interruption of the worker thread. worker.interrupt(); // Wait for the worker thread to finish. worker.join(); System.out.println("Main thread finished."); } } ``` 在这个例子中,主线程启动了一个子线程 (`worker`) 来完成一些任务。经过三秒钟之后,主线程请求子线程停止运行。子线程定期检查自己的中断状态并优雅地结束自己。 #### 3. **常见问题及其解决方案** ##### (1)**为什么我的线程没有响应 `interrupt()`?** 这通常是因为线程并未主动检查自身的中断状态或者正在进行不可中断的操作(比如长时间计算)。为了使线程能够及时响应中断信号,应在线程逻辑的关键位置加入对 `Thread.interrupted()` 或者 `Thread.currentThread().isInterrupted()` 的判断[^5]。 ##### (2)**恢复中断状态的重要性** 在捕获 `InterruptedException` 后重新调用 `Thread.currentThread().interrupt()` 是非常重要的实践之一。这是因为许多库函数依赖于这种机制来传递中断信息给更高级别的组件。 ##### (3)**处理拒绝策略中的中断** 在一个线程池环境中,当提交的任务超过容量限制时可能会触发拒绝策略,在此期间也可能涉及中断处理。例如,`ThreadPoolExecutor.AbortPolicy` 将直接抛出 `RejectedExecutionException` 而不尝试重试或将任务放入其他队列[^2]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值