线程的中断

1.应用场景:

假设从网络下载一个100M的文件,如果网速很慢,用户等得不耐烦,就可能在下载过程中点“取消”,这时,程序就需要中断下载线程的执行。

2.常用中断线程的方法:

1.使用标志位

2.interrupt()

(1)如果当前线程sleep,wait或因为其他线程使用join()而陷入阻塞状态,那么当在其他线程中调用了该线程的interrupt()方法时,join()等方法会立刻抛出InterruptedException异常,目标线程只要捕获到该异常,会立刻结束运行
 (2)interrupt()不能中断I/O阻塞和synchronized锁阻塞
 关于重置中断标志位:
isInterrupted()---不清除中断标志位,当前线程 
    public boolean isInterrupted() {//实例方法,不会重置标志位
        return isInterrupted(false);
    }
interrupted()-----清除中断标志位,又变成false,调用该方法的当前线程对象
    public static boolean interrupted() {//静态方法,会重置标志位
        return currentThread().isInterrupted(true);
    }

3.代码

3.1自己设置中断标志位

 class MyThreadTest extends Thread{
public  volatile boolean flag=true;
@Override
public void run() {

//1.标志位
while(flag){
        System.out.println(Thread.currentThread().getName()+"是否中断---"+
                        Thread.currentThread().isInterrupted()+
                        "------"+
                        Thread.currentThread().getState());
}
System.out.println("-----触发中断标志,中断了---");
}

//测试的代码

@Test
public void test1() throws InterruptedException {
    MyThreadTest t = new MyThreadTest("线程1");
    t.start();

    Thread.sleep(1);
    t.flag = false;
}

Java_16:线程中断_第1张图片

3.2调用 interrupt()

interrupted()

自定义线程测试
 class MyThread2 extends Thread {
@Override
public void run() {
                while (!isInterrupted()) {
                        System.out.println(Thread.currentThread().getName() + "是否中断---" +
                                        Thread.interrupted() +
                                        "------" +
                                        Thread.currentThread().getState());
                }
                System.out.println(Thread.currentThread().getName() + " end "+
                Thread.interrupted());
                System.out.println(Thread.currentThread().getName() + " end "+
                Thread.interrupted());

        }
    }

    //测试的代码

 @Test
public void test2() throws InterruptedException {
    MyThread2 t2=new MyThread2();
    t2.start();
    Thread.sleep(1);
    t2.interrupt();
}

Java_16:线程中断_第2张图片

直接在主线程测试

public static void main(String[] args) throws InterruptedException {

Thread.currentThread().interrupt();
System.out.println(Thread.interrupted());
System.out.println(Thread.interrupted());
}
}

Java_16:线程中断_第3张图片

isInterrupted()

自定义线程类

class MyThread3 extends Thread {
        @Override
        public void run() {
                while (!isInterrupted()) {
                        System.out.println(Thread.currentThread().getName() + "是否中断---" +
                                        Thread.currentThread().isInterrupted() +
                                        "------" +
                                        Thread.currentThread().getState());
                }
                System.out.println(Thread.currentThread().getName()+" end "
                +Thread.currentThread().isInterrupted());
                System.out.println(Thread.currentThread().getName()+" end "
                +Thread.currentThread().isInterrupted());
        }
}

 @Test
public void test3() throws InterruptedException {
        MyThread3 t3=new MyThread3();
        t3.start();
        Thread.sleep(1);
        t3.interrupt();
}

Java_16:线程中断_第4张图片

在主线程中测试

public static void main(String[] args) throws InterruptedException {
        Thread.currentThread().interrupt();
        System.out.println(Thread.currentThread().isInterrupted());
        System.out.println(Thread.currentThread().isInterrupted());
}
}

Java_16:线程中断_第5张图片

4.总结比较:

1.最好使用isInterrupted()方法进行中断标志的判断,因为不会重置标志位
2.对目标线程调用interrupt()方法可以请求中断一个线程,目标线程通过检测isInterrupted()标志获取自身是否已中断。
如果目标线程处于等待状态,该线程会捕获到InterruptedException;

3.目标线程检测到isInterrupted()为true或者捕获了InterruptedException都应该立刻结束自身线程;

4.通过标志位判断需要正确使用volatile关键字;

volatile关键字解决了共享变量在线程间的可见性问题。