美文网首页Java程序员Java学习笔记
Java多线程 - 如何正确的终止线程

Java多线程 - 如何正确的终止线程

作者: 嘉伟咯 | 来源:发表于2017-08-24 01:08 被阅读115次

最近打算读一下《实战java高并发程序设计》,夯实一下java多线程的知识。接下来应该会写一系列的读书笔记,里面会有多处引用到书中的代码或者文字。本文就是第一篇。

不推荐使用的stop方法

Thread.stop()是一个被废弃的方法,不被推荐使用的原因是stop方法太过于暴力,强行把执行到一半的线程终止,并且会立即释放这个线程所有的锁。会破坏了线程中引用对象的一致性。

例如在数据库中维护着一张用户表,记录了用户ID和用户名,使用Thread多线程写入两条记录:

记录1: ID=1,NAME=小明  
记录2: ID=2,NAME=小王

如果在记录1写到一半的时候被stop结束了,就可能出现各种奇怪的现象:

  1. 记录1被记录2覆盖,没有任何数据留下。
  2. 记录1只有一半,即只有ID,而NAME为空。
  3. 记录1和记录2混在同一条记录中,最后写入了一条一半是记录1一半是记录2的脏数据

所以,除非你很确定你在做什么,否则不要轻易使用stop方法

使用判断标志位的方法中断线程

那如果的确有中断线程的需求,我们需要怎么做呢?一般我们马上就会想到设置标志位的方法,即在线程中执行每一个步骤之前都判断一下是否需要退出线程:

class WorkThread extends Thread {
    private boolean mExitThread = false;

    public void exitThread() {
        mExitThread = true;
    }

    @Override
    public void run() {
        if (mExitThread) {
            return;
        }
        //do something

        if (mExitThread) {
            return;
        }
        //do something

        if (mExitThread) {
            return;
        }
        //do something
    }
}

其实Thread类早就帮我们实现了这个中断标志了。与Thread中断相关的方法有下面三个:

public void Thread.interrupt() //线程中断
public native boolean Thread.isInterrupted() //判断是否被中断
public static native boolean Thread.interrupted() //判断是否中断,并清除当前中断状态

所以上面的代码可以改写成这样:

class WorkThread extends Thread {
    @Override
    public void run() {
        if (isInterrupted()) {
            return;
        }
        //do something

        if (isInterrupted()) {
            return;
        }
        //do something

        if (isInterrupted()) {
            return;
        }
        //do something
    }
}

这个时候我们只需要调用Thread.interrupt()方法就能安全的中断线程了。

需要提醒一下的是Thread.interrupt()方法并不会像Thread.stop()方法一样立即结束线程,它只是设置了一个中断标志,需要在代码实现中去手动判断这个标志并且推出。

像下面这个代码就算掉了Thread.interrupt()方法也不会中断线程:

class WorkThread extends Thread {
    @Override
    public void run() {
        while(true){
            //do something
        }
    }
}

Thread.interrupt的优点

使用Thread.interrupt去中断线程除了可以免去自己实现标志位的烦恼之外,还可以中断sleep和wait中的线程。

还记得我们在调用Thread.sleep()这个方法的时候都需要catch或者抛出InterruptedException这个异常吗:

try {
    Thread.sleep(1000);
} catch (InterruptedException e) {
    e.printStackTrace();
}

这个InterruptedException异常就是由于我们调用了Thread.interrupt方法抛出的。所以Thread.interrupt可以打断Thread.sleep。同样Thread.wait也是可以被Thread.interrupt打断的。

需要注意的是如果sleep方法由于中断而抛出异常,此时,它会清除中断标记。所以在catch到这个异常的时候需要再次设置中断标记

Thread t1 = new Thread() {
    @Override
    public void run(){
        while(true){
            if(Thread.currentThread().isInterrupted()){
                break;
            }

            System.out.println("hello world");
            try{
                Thread.sleep(1000);
            }catch(InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        }
    }
};

相关文章

  • Java多线程 - 如何正确的终止线程

    最近打算读一下《实战java高并发程序设计》,夯实一下java多线程的知识。接下来应该会写一系列的读书笔记,里面会...

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

    如何正确停止Java线程,终止Java线程的三种方法 【原创,转载请注明出处】我的 【博客园主页】 【CSD...

  • Java多线程目录

    一 Java并发基础介绍二 Thread类三 线程终止四 Java多线程内存模型五 volatile关键字六 sy...

  • Java多线程编程核心技术

    第1章 Java多线程技能 在Java中有以下3种方法可以终止正在运行的线程:1)使用退出标志,使线程正常退出,也...

  • Java多线程目录

    Java多线程目录 Java多线程1 线程基础Java多线程2 多个线程之间共享数据Java多线程3 原子性操作类...

  • 带你搞懂Java多线程(五)

    带你搞懂Java多线程(一)带你搞懂Java多线程(二)带你搞懂Java多线程(三)带你搞懂Java多线程(四) ...

  • 带你搞懂Java多线程(六)

    带你搞懂Java多线程(一)带你搞懂Java多线程(二)带你搞懂Java多线程(三)带你搞懂Java多线程(四)带...

  • (1)如何开启多线程

    前言 学习java多线程首先就是要知道如何开启多线程.在java中开启多线程主要有两种方式: 继承Thread类,...

  • 第四章 并发编程基础

    线程的状态。如何启动和终止线程。多线程之间通信的基本方式。多线程编程的等待/通知的经典范式。 现代操作系统调度的最...

  • Java中如何正确的终止线程

    1、Java中API自带的stop()方法,来终止线程,不推荐。 stop()方法是一个被废弃的方法。为什么sto...

网友评论

    本文标题:Java多线程 - 如何正确的终止线程

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