package com.mozq.thread.interrupt; /** * 注意:调用interrupt()方法,并不会结束线程。 * 结束线程的语义:需要我们自己使用3个中断方法构建。 * * 没有任何语言方面的需求一个被中断的线程应该终止。 * 中断一个线程只是为了引起该线程的注意,被中断线程可以决定如何应对中断。 * 某些线程非常重要,以至于它们应该不理会中断,而是在处理完抛出的异常之后继续执行。 * 但是更普遍的情况是,一个线程将把中断看作一个终止请求,这种线程的run方法遵循如下形式 public void run() { try { ... * 不管循环里是否调用过线程阻塞的方法如sleep、join、wait,这里还是需要加上 * !Thread.currentThread().isInterrupted()条件,虽然抛出异常后退出了循环,显 * 得用阻塞的情况下是多余的,但如果调用了阻塞方法但没有阻塞时,这样会更安全、更及时。 while (!Thread.currentThread().isInterrupted()&& more work to do) { do more work } } catch (InterruptedException e) { //线程在wait或sleep期间被中断了 } finally { //线程结束前做一些清理工作 } } 上面是while循环在try块里,如果try在while循环里时,因该在catch块里重新设置一下中断标示, 因为抛出InterruptedException异常后,中断标示位会自动清除,此时应该这样: public void run() { while (!Thread.currentThread().isInterrupted()&& more work to do) { try { ... sleep(delay); } catch (InterruptedException e) { Thread.currentThread().interrupt();//重新设置中断标示 } } } * 结束线程方法2:interrupt()方法,interrupted()方法和isInterrupted()方法 * 该方法的中断语义详细定义在说明文档中。和通常所说的中断意思不一致。 * interrupt()方法用于设置: * 如果线程处于wait,sleep,join阻塞,则会发生中断异常。中断状态为被清除。 * 一般情况则导致中断状态位被设置 * isInterrupted()检测不清除 * interrupted()检测并清除。 * 通过这3个方法,适用于一个线程向另一个线程发送信息,并被B线程处理。 * 可以用来实现线程A根据情况结束线程B的语义。 * * 分成2种情况,1.线程处于阻塞状态2.线程处于正常运行 * 线程处于正常运行 * public class ThreadSafe extends Thread { public void run() { while (!isInterrupted()){ //do something, but no throw InterruptedException } } } 线程处于阻塞状态 public class ThreadSafe extends Thread { public void run() { while (!isInterrupted()){ //非阻塞过程中通过判断中断标志来退出 try{ Thread.sleep(5*1000);//阻塞过程捕获中断异常来退出 }catch(InterruptedException e){ e.printStackTrace(); break;//捕获到异常之后,执行break跳出循环。 } } } } 1 * @author jie * */ public class MyInterruptMethod { public static void main(String[] args) { Thread t = new Thread() { //方法1: /* @Override public void run() { try { while(!Thread.currentThread().isInterrupted()) { System.out.println(Thread.currentThread().getName() + "...run"); Thread.sleep(100); } } catch (InterruptedException e) {} } */ @Override public void run() { while(!Thread.currentThread().isInterrupted()) { System.out.println(Thread.currentThread().getName() + "...run"); try { Thread.sleep(10); } catch (InterruptedException e) { //因为虚拟机处理interrupt调用时,如果方法处于阻塞,会抛出异常并清除中断标志,所以此处需要重置。 Thread.currentThread().interrupt(); //return; //break; } } } }; t.start(); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } //业务逻辑。。。 //此处想结束线程 t.interrupt(); } }