java 停止线程的正确方式

在看一个问题的时候突然想到之前面试,有一个面试官问我,怎么停止线程,突然断片了,趁着现在有时间,做一下总结。

在网上看了几篇文章,我这篇文章大致记录一下,

1、第一种方式:使用stop方法终止线程

这种方式最直接了当,但是也是不可取的,调用stop()方法时会抛出java.lang.ThreadDeath异常,但是通常情况下,此异常不需要显示地捕捉

public class MyThread extends Thread {
    private int i = 0;
    public void run(){
        super.run();
        try {
            this.stop();
        } catch (ThreadDeath e) {
            System.out.println("进入异常catch");
            e.printStackTrace();
        }
    }
}

public class Run {
    public static void main(String args[]) throws InterruptedException {
        Thread thread = new MyThread();
        thread.start();
    }
}

程序中可以直接使用thread.stop()来强行终止线程,但是stop方法是很危险的,就象突然关闭计算机电源,而不是按正常程序关机一样,可能会产生不可预料的结果,不安全主要是:thread.stop()调用之后,创建子线程的线程就会抛出ThreadDeatherror的错误,并且会释放子线程所持有的所有锁。一般任何进行加锁的代码块,都是为了保护数据的一致性,如果在调用thread.stop()后导致了该线程所持有的所有锁的突然释放(不可控制),那么被保护数据就有可能呈现不一致性,其他线程在使用这些被破坏的数据时,有可能导致一些很奇怪的应用程序错误。因此,并不推荐使用stop方法来终止线程。

2、第二种方式 设置标识位:

就是继承线程 添加自己的成员变量,当变量状态改变时,停止线程:

public class ThreadTest extends Thread {
///使用volatile目的是保证可见性,一处修改了标志,处处都要去主存读取新的值,而不是使用缓存  
    public volatile boolean exit = false;

    public void run() {
        while (!exit) {
            for (int i = 1; i < 100; i++) {
                System.out.print(">>>>:i:" + i + "\n");
            }
        }
    }
}

3、第三种方式:使用interrupt()中断的方式

3.1 interrupt()方法的使用效果并不像for+break语句那样,马上就停止循环。调用interrupt方法是在当前线程中打了一个停止标志,并不是真的停止线程。

public class MyThread extends Thread {
    public void run(){
        super.run();
        for(int i=0; i<500000; i++){
            System.out.println("i="+(i+1));
        }
    }
}

public class Run {
    public static void main(String args[]){
        Thread thread = new MyThread();
        thread.start();
        try {
            Thread.sleep(2000);
            thread.interrupt();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

输出结果:

...
i=499994
i=499995
i=499996
i=499997
i=499998
i=499999
i=500000

3.2 判断线程是否停止状态

Thread.java类中提供了两种方法:

  • this.interrupted(): 测试当前线程是否已经中断;

内部实现是调用的当前线程的isInterrupted(),并且会重置当前线程的中断状态。由该方法清除如果线程中断,第一次返回true 后修改中断状态,再次调用返回false,如果连续两次调用该方法,则第二次调用返回false。

  • this.isInterrupted(): 测试线程是否已经中断;

isInterrupted()是实例方法,是调用该方法的对象所表示的那个线程的isInterrupted(),不会重置当前线程的中断状态。

3.3 使用interrupt()方法来中断线程有两种情况

1> 线程阻塞状态

如使用了sleep,同步锁的wait,socket中的receiver,accept等方法时,会使线程处于阻塞状态。

当调用线程的interrupt()方法时,会抛出InterruptException异常。阻塞中的那个方法抛出这个异常,通过代码捕获该异常,然后break跳出循环状态,从而让我们有机会结束这个线程的执行。通常很多人认为只要调用interrupt方法线程就会结束,实际上是错的, 一定要先捕获InterruptedException异常之后通过break来跳出循环,才能正常结束run方法。

public class ThreadTest extends Thread {
    public void run() { 
        while (true){
            try{
                    Thread.sleep(5*1000);//阻塞5妙
                }catch(InterruptedException e){
                    e.printStackTrace();
                    break;//捕获到异常之后,执行break跳出循环。
                }
        }
    } 
}

2>线程未处于阻塞状态

使用isInterrupted()判断线程的中断标志来退出循环。当使用interrupt()方法时,中断标志就会置true,和使用自定义的标志来控制循环是一样的道理。 

public class ThreadTest extends Thread {
    public void run() { 
        while (!isInterrupted()){
            //do something, but no throw InterruptedException
        }
    } 
}

为什么要区分进入阻塞状态和和非阻塞状态两种情况了,是因为当阻塞状态时,如果有interrupt()发生,系统除了会抛出InterruptedException异常外,还会调用interrupted()函数,调用时能获取到中断状态是true的状态,调用完之后会复位中断状态为false,所以异常抛出之后通过isInterrupted()是获取不到中断状态是true的状态,从而不能退出循环,因此在线程未进入阻塞的代码段时是可以通过isInterrupted()来判断中断是否发生来控制循环,在进入阻塞状态后要通过捕获异常来退出循环。因此使用interrupt()来退出线程的最好的方式应该是两种情况都要考虑:

public class ThreadTest extends Thread {
    public void run() { 
        while (!isInterrupted()){ //非阻塞过程中通过判断中断标志来退出
            try{
                Thread.sleep(5*1000);//阻塞过程捕获中断异常来退出
            }catch(InterruptedException e){
                e.printStackTrace();
                break;//捕获到异常之后,执行break跳出循环。
            }
        }
    } 
}

总结

对于线程的终止,个人觉得用得最多也最安全的还是属于第二种使用volatile标志位来终止线程,因为第三种的使用受到很多方面的制约,一旦没用好会出各种问题。

得看你的具体业务场景,如果是标志位中断的话,标志位的位置安放也是要注意的,如果你用到的是第三种interrupt中断线程就得综合考虑线程所处的状态,如果抛异常那么这个异常抛出来的之后的代码处理也是正确中断线程的关键。
 

你可能感兴趣的:(java,面试系列)