美文网首页
结束线程

结束线程

作者: TTTqiu | 来源:发表于2016-11-13 14:50 被阅读18次
  • 线程对象属于一次性消耗品,一般线程执行完run方法之后,线程就正常结束了,线程结束之后就报废了,不能再次start,只能新建一个线程对象。
  • 但有时run方法是永远不会结束的。例如在程序中使用线程进行Socket监听请求,或是其他的需要循环处理的任务。在这种情况下,一般是将这些任务放在一个循环中,如while循环。当需要结束线程时,如何退出线程呢?
    有三种方法可以结束线程:
  1. 使用退出标志,使线程正常退出,也就是当run方法完成后线程终止
  2. 使用interrupt()方法中断线程
  3. 使用stop方法强行终止线程(不推荐使用,可能发生不可预料的结果)
    前两种方法都可以实现线程的正常退出,也就是要谈的优雅结束线程;第3种方法相当于电脑断电关机一样,是不安全的方法。

1. 使用退出标志终止线程

使用一个变量来控制循环,例如最直接的方法就是设一个boolean类型的标志,并通过设置这个标志为true或false来控制while循环是否退出。代码如下:

public class ThreadSafe extends Thread {
    public boolean exit = false; 
    public void run() { 
        while (!exit){
            //do something
        }
    } 
}

定义了一个退出标志exit,当exit为true时,while循环退出,exit的默认值为false.在定义exit时,使用了一个Java关键字volatile,这个关键字的目的是使exit同步,也就是说在同一时刻只能由一个线程来修改exit的值,

2. 使用interrupt()方法终止线程

使用interrupt()方法来终端线程可分为两种情况:

  1. 线程处于阻塞状态
    如使用了sleep,同步锁的wait,socket的receiver,accept等方法时,会使线程处于阻塞状态。如果在这时调用了线程的interrupt()方法,系统会抛出一个InterruptedException异常,代码中通过捕获异常,然后break跳出循环状态,使线程正常结束。
    通常很多人认为只要调用interrupt方法线程就会结束,实际上是错的,一定要先捕获InterruptedException异常之后通过break来跳出循环,才能正常结束run方法。
    (当然如果try在while外面,就不用break了。捕捉到异常就会结束线程。)
public class ThreadSafe extends Thread {
    public void run() { 
        while (true){
            try{
                Thread.sleep(5*1000);阻塞5妙
            }catch(InterruptedException e){
                e.printStackTrace();
                break;//捕获到异常之后,执行break跳出循环。
            }
        }
    } 
}
  1. 线程未进入阻塞状态
    使用isInterrupted()判断线程的中断标志来退出循环,当调用了interrupt()方法时,中断标志就会置true,和使用自定义的标志来控制循环是一样的道理。
public class ThreadSafe extends Thread {
    public void run() { 
        while (!isInterrupted()){
            //do something, but no tthrow InterruptedException
        }
    } 
}

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

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

然后在需要停止线程的地方调用threadSafe.interrupted()就能终止线程了。

3. 使用stop()方法终止线程(不推荐)

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

相关文章

  • 结束线程

    线程对象属于一次性消耗品,一般线程执行完run方法之后,线程就正常结束了,线程结束之后就报废了,不能再次start...

  • Java19守护线程

    在程序后台执行的线程,当程序结束时,守护线程也结束。程序结束:所有的非守护线程结束。

  • Kotlin线程(4)线程管理

    等待线程结束 (join)线程让步 (yield)线程停止 一、等待线程结束   当一个线程调用其它线程(t1)的...

  • 多线程练习

    一个进程结束的前提为进程内所有线程结束。结束子线程-->让子线程中的任务结束(就是让run方法结束:结束循环多用添...

  • 终止线程方式

    正常运行结束 程序运行结束,线程自动结束。 使用退出的标志退出线程 一般 run()方法执行完,线程就会正常结束,...

  • 线程-11.24

    1.使用threading模块 单线程执行: 多线程执行: 主线程会等待所有的子线程结束后才结束 查看线程数量: ...

  • 4.0.6.守护线程,线程死锁

    守护线程会随着主线程的结束而结束DaemonThread 线程 1, 线程 2,锁1,锁2 线程1 有锁1,想拿锁...

  • java多线程

    1.线程分用户线程和守护线程 用户线程全部结束时虚拟机和守护线程也准备即结束 守护线程代表如GC线程 2.sync...

  • 守护线程(后台线程)

    与前台线程的区别: 开启、运行都没有区别,只有结束有区别:要前台线程结束后,后台线程才会结束,而且是自动结束的。

  • 多线程爬虫之糗事百科(二)

    在python3中,主线程结束,子线程不会结束 在python3中,主进程结束,子进程不会结束 为了能够让主线程回...

网友评论

      本文标题:结束线程

      本文链接:https://www.haomeiwen.com/subject/ozdcpttx.html