如何优雅地中止线程?

栏目: IT技术 · 发布时间: 4年前

内容简介:本文来学习如何学习优雅的中止线程?通过Java 线程的生老病死的学习,我相信大家对线程的运行以及线程的状态有一定了解了,那么我们现在来学习中止线程:首先来讲解一个错误的方式来中止线程 —接下来通过一段程序来讲解为什么

本文来学习如何学习优雅的中止线程?通过 Java 线程的生老病死的学习,我相信大家对线程的运行以及线程的状态有一定了解了,那么我们现在来学习中止线程:

错误的线程中止 - stop

首先来讲解一个错误的方式来中止线程 — stop :中止线程,并且清除监控器锁的信息,但是可能导致线程安全问题,JDK 不建议使用,类似的方法还有 destory,由于 JDK 从未实现该方法,在这里就不介绍了。

接下来通过一段程序来讲解为什么 stop 会导致线程安全问题?

首先定义一个线程类 StopThread

public class StopThread extends Thread {
    private int i = 0;
    private int j = 0;

    @Override
    public void run() {
        synchronized (this) {
            // 增加同步锁,确保线程安全
            ++i;
            try {
                // 休眠10秒,模拟耗时操作
                Thread.sleep(10000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            ++j;
        }
    }

    /**
     * 打印 i 和 j
     */
    public void print() {
        System.out.println("i=" + i + " j=" + j);
    }
}

这个线程做的事情就是在同步代码块中对 ij 这两个变量进行自增操作,但是在这个执行过程中会进行 10 秒的睡眠,如果在这个过程中,如果用 stop 方法将线程中止的话,会导致 ij 数据不正确,也可以说程序设计上的线程安全问题,因为主线程影响到了创建的 StopThread 线程的数据不正确性,理想的正确输出结果应该是要么全部添加成功,要么都失败,因为我们添加锁的目的就是保证操作原子性或者说想让这两个变量在操作的时候不受其他线程干扰。

下面编写 StopThreadDemo 类,来使用 stop 方法做个错误示范:

public class StopThreadDemo {
    public static void main(String[] args) throws InterruptedException {
        StopThread thread = new StopThread();
        thread.start();
        // 休眠 1 秒,确保 i 变量自增成功
        Thread.sleep(1000);
        // 暂停线程
        thread.stop(); // 错误的终止
        while (thread.isAlive()) {
            // 确保线程已经终止
        } // 输出结果
        thread.print();
    }
}

StopThreadDemo 类中,创建并启动了 StopThread 线程,这个线程就是下执行变量 ij 的自增操作,但是这个自增操作是用同步关键字包裹的同步代码块,这样做是为了让两个变量的自增操作实现 原子性 ,不会受到其他线程的干扰,确保线程的安全,但是在线程休眠的 10 秒内,通过 stop 方法把线程中止掉,会发现输出结果为 i=1 j=0 ,也就是代码的前半段 i 自增实现,但是后半段 j 的自增失败,会使线程中的数据出现不一致性,也就是同步代码块的保证的原子性的目标没有达成,破坏了线程安全。

正确的线程中止 - interrupt

在介绍了错误的中止方式后,让我们来学习正确的线程中止 - interrupt

如果目标线程在调用 Object classwait() 、wait(long) 或 wait(long,int) 方法、join()、join(long,int) 或 sleep(long,int) 方法时阻塞,那么 interrupt 会生效,该线程的中断状态将被清除,抛出 InterruptedException 异常。

如果目标线程是被 IO 或者 NIO 中的 Channel 所阻塞,同样 IO 操作会被中断返回特殊异常值,达到中止线程的目的。

如果以上条件都不满足,则会设置此线程的中断状态。

接下来将 StopThreadDemo 中的 stop 改为 interrupt 来看下运行结果是什么:

java.lang.InterruptedException: sleep interrupted
	at java.lang.Thread.sleep(Native Method)
	at com.wupx.thread.StopThread.run(StopThread.java:18)
i=1 j=1

可以发现两个变量的自增可以正常执行,保证了执行的数据一致性, interrupt 不会强制中止,将线程直接中断,而是抛出异常通知我们,开发者就可以控制收到异常后的执行逻辑,让整个程序处于线程安全的状态,这是目前 JDK 版本中推荐的 interrupt 方法。

除了 interrupt 的正确方法外,还可以通过标志位的形式来中止线程:

正确的线程中止 - 标志位

如果代码程序逻辑中是循环执行的业务,可以在程序的执行中线程代码中增加一个标志位,比如下面代码中在 while 循环中去执行这个程序,通过 flag 去控制程序是否继续执行,如果在外部线程将 flag 修改为 false ,那么创建的子线程代码中会收到这个数据的变化,通过这个变量的形式,通知到另一个线程,从而达到控制线程中止的效果。

public class FlagThreadDemo {
    public volatile static boolean flag = true;

    public static void main(String[] args) throws InterruptedException {
        new Thread(() -> {
            try {
                while (flag) { // 判断是否运行
                    System.out.println("运行中");
                    Thread.sleep(1000L);
                }
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }).start();
        // 3 秒之后,将状态标志改为 false,代表不继续运行
        Thread.sleep(3000L);
        flag = false;
        System.out.println("程序运行结束");
    }
}

通过运行代码,得到的结果如下:

运行中
运行中
运行中
程序运行结束

这种方式受限于线程中所执行的业务逻辑,如果程序中是有可以用来做标志位的条件的话可以用这种方式来做,也是一种正确的线程中止方式。

总结

本文主要讲解了线程中止的三种方式: stopinterrupt 以及标志位,大家学会了吗,欢迎留言讨论。

源代码可以在公众号【 武培轩 】中回复【 并发 】获取。


以上所述就是小编给大家介绍的《如何优雅地中止线程?》,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对 码农网 的支持!

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

数据结构与算法

数据结构与算法

BrunoRPreiss / 电子工业出版社 / 2003-1 / 55.00元

本书是作者根据他在滑铁卢大学计算机工程学院教授数据结构与算法课程的经验编写而成的。它采用C++面向对象的设计模式,不仅系统全面地介绍了各种传统的数据结构,还把它们按照类和类层次的现代理念予以展开,进而达到抽象结构与实际设计的完美统一。本书的后三章通过引入抽象问题求解的概念,集中讲述了算法技术和各算法之间的关系。另外,作者运用一定的数学工具以及必要的分析技术和分析理论,对每种数据结构及相关算法都进行一起来看看 《数据结构与算法》 这本书的介绍吧!

图片转BASE64编码
图片转BASE64编码

在线图片转Base64编码工具

HTML 编码/解码
HTML 编码/解码

HTML 编码/解码

XML 在线格式化
XML 在线格式化

在线 XML 格式化压缩工具