如何正确停止Java线程,终止Java线程的三种方法

御弟哥哥
• 阅读 1472

如何正确停止Java线程,终止Java线程的三种方法

在 Java 中有以下 3 种方法可以终止正在运行的线程:

  1. 使用退出标志,使线程正常退出,也就是当 run() 方法完成后线程终止。
  2. 使用 stop() 方法强行终止线程,但不推荐,该方法已被弃用,原因见后文。
  3. 使用 interrupt 方法中断线程。

以下内容翻译自 JDK1.5官方文档 ,内容有微调。未发现JDK1.8对应文档与JDK1.5的内容有明显不同。

停止一个线程的推荐做法

stop的大多数用法应由简单地修改某些变量以指示目标线程应停止运行的代码代替。 目标线程应定期检查此变量,如果该变量指示要停止运行,则应有序地从其运行方法返回。 (这是Java始终推荐的方法)。为了确保对stop-request进行及时的通信,变量必须是volatile(或必须同步访问变量)

例如,假设您的程序包含以下startstoprun方法:

 private Thread blinker;

    public void start() {
        blinker = new Thread(this);
        blinker.start();
    }

    public void stop() {
        blinker.stop();  // UNSAFE!
    }

    public void run() {
        Thread thisThread = Thread.currentThread();
        while (true) {
            try {
                thisThread.sleep(interval);
            } catch (InterruptedException e){
            }
            repaint();
        }
    } 

可以通过将程序的stop和run方法替换为下面的代码来避免使用Thread.stop

 private volatile Thread blinker;

    public void stop() {
        blinker = null;
    }

    public void run() {
        Thread thisThread = Thread.currentThread();
        while (blinker == thisThread) {
            try {
                thisThread.sleep(interval);
            } catch (InterruptedException e){
            }
            repaint();
        }
    } 


=== 后面的内容可以不看,看的话务必从头到尾按顺序阅读 ===



为什么 Thread.stop 被废弃?

因为它本质上是不安全的。 _停止线程会使它解锁它已锁定的所有监视器_。 (当ThreadDeath异常在堆栈中向上传播时,监视器将被解锁。)如果先前由这些监视器保护的任何对象处于不一致状态,则其他线程现在可能会以不一致状态查看这些对象。 这样的对象被称为_已损坏的对象_。 当线程对损坏的对象进行操作时,可能会导致任意行为。 此行为可能是微妙的,难以检测,或者可能是明显的。 与其他未检查的异常不同,ThreadDeath会无声地杀死线程。 因此,用户没有警告其程序可能已损坏。 在实际损坏发生后的任何时间,腐败会体现出来(注:corruption,意为腐败,类似代码的bad smell,指程序中出现的问题或bug)。

不能只是捕捉ThreadDeath异常并修复损坏的对象吗?

从理论上讲,也许可以,但这会_使编写正确的多线程代码的任务大大复杂化_。该任务几乎是无法克服的,原因有两个:

  1. 线程_几乎可以在任何地方_引发ThreadDeath异常 。考虑到这一点,必须仔细研究所有同步的方法和块。
  2. ThreadDeath从第一个(在catchor finally子句中)清除时,线程可以引发第二个异常。必须重复进行清理,直到成功。确保该代码很复杂。

如何停止等待较长时间的线程(例如,等待输入)?

这就是Thread.interrupt方法的用途。 可以使用上面展示的相同的“基于状态”的信令机制,但是状态更改(在前面的示例中,_blinker = null_)之后可以调用Thread.interrupt来中断等待:

 public void stop() {
        Thread moribund = waiter;
        waiter = null;
        moribund.interrupt();
    } 

为了使该技术起作用,至关重要的是,任何捕获中断异常并且不准备处理中断异常的方法都必须立即重新声明该异常。 我们说重新声明而不是重新抛出,因为并非总是可能重新抛出异常。 如果未声明捕获InterruptedException的方法引发此(checked)Exception,则它应使用下面的提示“重新中断自身”:

Thread.currentThread().interrupt(); 

这样可以确保线程尽快引发InterruptedException

如果线程不响应Thread.interrupt怎么办?

在某些情况下,您可以使用特定于应用程序的技巧。 例如,如果某个线程正在一个已知的套接字上等待,则可以关闭该套接字以使该线程立即返回。 不幸的是,一般来说,真的没有什么技术能奏效。 应该注意的是,在所有等待线程不响应Thread.interrupt的情况下,它也不响应Thread.stop 这样的情况包括故意的拒绝服务攻击,以及Thread.stopThread.interrupt无法正常工作的I / O操作。(_注释_:这段的个人理解,如果程序不响应Thread.interrupt,那就没办法了,自己去想别的方法解决吧。所以还是尽量使用推荐的方法来控制线程停止与运行)

为什么不赞成使用Thread.suspendThread.resume

Thread.suspend本质上容易死锁。 如果目标线程在挂起时,在监视器上持有一个保护关键系统资源的锁,则在恢复目标线程之前,没有线程可以访问该资源。 如果将恢复目标线程的线程在调用resume之前尝试锁定此监视器,则会导致死锁。 这种僵局通常表现为“冻结”进程。

应该使用什么代替Thread.suspendThread.resume

Thread.stop一样,谨慎的方法是让“目标线程”轮询一个指示线程所需状态(活动或挂起)的变量。 当所需的状态被挂起时,线程使用Object.wait等待。 恢复线程后,将使用Object.notify通知目标线程。

例如,假设您的程序包含以下“鼠标按下”事件处理程序,该事件处理程序的功能是可以切换blinker线程的状态:

 private boolean threadSuspended;

    Public void mousePressed(MouseEvent e) {
        e.consume();

        if (threadSuspended)
            blinker.resume();
        else
            blinker.suspend();  // DEADLOCK-PRONE!容易出现死锁

        threadSuspended = !threadSuspended;
    } 

您可以通过将上面的事件处理程序替换为下面的代码来避免使用Thread.suspendThread.resume

 public synchronized void mousePressed(MouseEvent e) {
        e.consume();

        threadSuspended = !threadSuspended;

        if (!threadSuspended)
            notify();
    } 

并添加下面的代码到 "run loop":

 synchronized(this) {
        while (threadSuspended)
            wait();
    } 

wait方法抛出InterruptedException,因此它必须在try ... catch子句中。 可以将其与sleep放在同一个的子句中。 该检查应在sleep之后(而不是在sleep之前),以便在线程“resumed”时立即重新绘制窗口。 生成的run方法如下:

 public void run() {
        while (true) {
            try {
                Thread.currentThread().sleep(interval);

                synchronized(this) {
                    while (threadSuspended)
                        wait();
                }
            } catch (InterruptedException e){
            }
            repaint();
        }
    } 

请注意,“鼠标按下”方法中的notify和run方法中的wait在同步块(synchronized)内部。这种做法是编程语言要求的,并确保wait和notify正确地按顺序执行。 实际上,这消除了可能导致“已暂停”线程错过通知并无限期保持暂停的竞争条件。

随着平台的成熟,尽管Java同步(synchronized)的性能开销在降低,但它永远不会“免费”(注:免费指几乎可以忽略的性能开销)。 一个简单的技巧可以用来删除我们添加到“运行循环”的每个迭代中的同步。 所添加的同步块被稍微复杂一点的代码所代替,仅当线程实际上已被挂起时才进入同步块:

 if (threadSuspended) {
        synchronized(this) {
            while (threadSuspended)
                wait();
        }
    } 

在没有显式同步的情况下,必须将threadSuspended设置为volatile,以确保及时传达suspend-request。

最终生成的run方法为:

 private boolean volatile threadSuspended;

    public void run() {
        while (true) {
            try {
                Thread.currentThread().sleep(interval);

                if (threadSuspended) {
                    synchronized(this) {
                        while (threadSuspended)
                            wait();
                    }
                }
            } catch (InterruptedException e){
            }
            repaint();
        }
    } 

可以结合两种技术来产生可以安全地“停止”或“暂停”的线程吗?

可以的, 这相当简单。 一个微妙之处是目标线程可能在另一个线程试图将其停止时已被挂起。 如果stop方法仅将状态变量(blinker)设置为null,则目标线程将保持挂起状态(在监视器上等待),而不是应有的退出。 如果重新启动了applet,则多个线程可能最终会同时在监视器上等待,从而导致行为不稳定。

要纠正这种情况,stop方法必须确保目标线程在挂起后立即恢复。 目标线程恢复后,必须立即识别出它已停止,并正常退出。 这是生成的run和stop方法的外观:

 public void run() {
        Thread thisThread = Thread.currentThread();
        while (blinker == thisThread) {
            try {
                thisThread.sleep(interval);

                synchronized(this) {
                    while (threadSuspended && blinker==thisThread)
                        wait();
                }
            } catch (InterruptedException e){
            }
            repaint();
        }
    }

    public synchronized void stop() {
        blinker = null;
        notify();
    } 

如上所述,如果stop方法调用Thread.interrupt,它也不必调用notify,但仍必须同步。 这样可以确保目标线程不会由于竞争条件而错过中断。

关于 Thread.destroy 方法

Thread.destroy从未真正被实现。 如果实现了它,使用Thread.suspend容易发生死锁。 (实际上,它与Thread.suspend大致等效,没有后续的Thread.resume的可能性。)我们目前不实现它,但也不会弃用它(将来将阻止其实现)。 尽管肯定会发生死锁,但有人认为,在某些情况下,某个程序愿意冒着deadlock的风险也不想直接退出。

Java8中的destroy方法源码:

 /**
     * Throws {@link NoSuchMethodError}.
     * 此方法最初设计为在不进行任何清理的情况下销毁此线程。
     * 它所持有的任何监视器都将保持锁定状态。
     * 但是,该方法从未实现。如果要实现,则将以{@link #suspend}的方式发生死锁。
     * 如果目标线程在被销毁时持有一个保护关键系统资源的锁,则没有线程可以再次访问该资源。 
     * 如果另一个线程曾尝试锁定此资源,则会导致死锁。这种死锁通常表现为“冻结”进程。
     */
    @Deprecated
    public void destroy() {
        throw new NoSuchMethodError();
    } 
点赞
收藏
评论区
推荐文章
待兔 待兔
5个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Wesley13 Wesley13
3年前
java多线程方法笔记【上】
刚开始工作,主要使用c写服务器,不过偶尔也会用到java和python,还是学一学吧。先从java多线程开始。java多线程的基本方法://线程开始运行 public void start( );  //线程运行的方法runpublic void run( );  //线程休眠,单位毫秒 
Wesley13 Wesley13
3年前
java多线程之停止线程
在多线程开发中停止线程是很重要的技术点。停止线程在Java语言中并不像break语句那样干脆,需要一些技巧性的处理。一、 异常法采用异常法来停止一个线程,首先我们需要了解一下两个方法的用法:1、interrupt()方法publicclassMyTh
Wesley13 Wesley13
3年前
Java高并发编程四
_做个笔记,java线程常用的方法,耐心看完._编号方法说明1publicvoidstart()使该线程开始执行;Java虚拟机调用该线程的run方法。2publicvoidrun()如果该线程是使用独立的Runnable运行对象构造的,则调用该Runnable对象的run方法;否则,该方
Wesley13 Wesley13
3年前
Java面试题速查手册
Java类equals抽象类类型StringIntegershort多线程概念锁synchronized线程创建及状态线程通讯线程安全ThreadLocalatomicvolatile终止线程反射什么是反射?序列化什么是java序
Wesley13 Wesley13
3年前
Java多线程学习笔记
Java中创建多线程的三种方法1、继承Thread类创建线程2、实现Runnable接口创建线程3、使用Callable和Future创建线程\
Wesley13 Wesley13
3年前
Java线程停止方法之Interrupt方法
  最近在学习Java多线程相关的知识点,其中关于线程停止的方法网上也有不少大牛给出了详细的解答,而我这边就其中Interrupt方法的注意点给自己提个醒。  首先还是大概的罗列下停止线程的方法:  1、使用stop()方法等,不过已经不再被推荐使用,和suspend、resume一样。  2、使用退出标志终止线程,引入一个共享变量,volati
Wesley13 Wesley13
3年前
2.Java 并行程序基础
1.初始线程:线程的基本操作1.新建线程2.终止线程stop造成数据不一致3.线程中断publicvoidThread.interrupt()//中断线程publicbooleanThread.isTnterrup
Wesley13 Wesley13
3年前
C# 线程基础
1 线程是进程中的一个执行流 2线程是一个可以单独操作的活动3线程创建和常用方法 a 创建    Thread thnewThread(Method); b常见方法 th.start()//启动线程 th.Abort()//终止线程 Thread.Sleep(n)//休眠线程(停止n毫秒后继续执
小万哥 小万哥
9个月前
深入理解 Java 多线程、Lambda 表达式及线程安全最佳实践
Java线程线程使程序能够通过同时执行多个任务而更有效地运行。线程可用于在不中断主程序的情况下在后台执行复杂的任务。创建线程有两种创建线程的方式。1.扩展Thread类可以通过扩展Thread类并覆盖其run()方法来创建线程:javapublicclas