[Java多线程编程之三] 线程中止的多种姿势

一、错误的姿势- Stop

  • Thread.stop()
1、存在问题

  不管程序的运行逻辑如何,stop会直接中止线程,并清除监控器锁的信息,如果有些代码块的运行具有原子性,则stop可能会破坏这种原子性导致线程安全问题,所以JDK已不建议使用。

2、代码示例

  定义一个Thread的子类StopThread,在重写的run()中,++i++j被包裹在同一同步代码块中,目的是让i和j同时加1,加锁保证同一时刻只有一个线程可以执行同步代码块,保证原子性,在理想状况下,i应该是永远等于j。

public class StopThread extends Thread {
    private int i = 0, j = 0;
    
    @Override
    public void run() {
        synchronized (this) {
            // 增加同步锁,确保线程安全
            ++i;
            try {
                // 休眠10秒,模拟耗时操作
                Thread.sleep(10000L);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            ++j;
        }
    }
    
    // 打印i和j
    public void print() {
        System.out.println("i=" + i + " j=" + j);
    }
}

  但是如果在线程休眠的时候,调用了stop方法,这时i已经执行了加1操作,j尚未执行,由于线程会直接释放锁并不再执行后面的代码,会导致i和j加1操作的次数不相等,i跟j不相等,发生线程安全问题。

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

  程序执行结果:

i=1 j=0


二、正确的姿势- interrupt

  • Thread.interrupt()
      interrupt()不会粗暴地将代码执行从中间拦腰折断,而是会让被中断的线程先执行完代码,然后再抛出一个中断异常让外部程序捕获或者改变线程的状态让外部程序判断。
1、调用interrupt()导致抛出InterruptedException的情况,同时清理目标线程的中断状态
  • 目标线程正处于调用了wait()、wait(long)、wait(long, int)后的WAITING线程状态
  • 目标线程正处于调用了join()、join(long, int) 后的WAITING的线程状态
  • 目标线程正处于调用了sleep(long, int)后的BLOCKED的线程状态
2、中断或返回异常值的情况

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

3、其他情况

  设置线程的中断状态

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

  程序执行结果:

java.lang.InterruptedException: sleep interrupted
i=1 j=1
    at java.lang.Thread.sleep(Native Method)
    at chapter1.section3.ThreadStop.StopThread.run(StopThread.java:13)


三、适用于简单程序的姿势 - 标志位

1、用法

  一般用volatile修饰定义标志位,使其具有线程可见性;控制线程可通过控制标志位的值来控制目标线程;目标线程在其执行程序内自旋循环判断标志位的值,当值被切换时,跳出循环执行对应分支的程序。

2、代码示例

  如下面的代码所示,主线程修改flag的值,让匿名线程中止,通过控制标志位flag的值,达到控制匿名线程程序运行的效果。

public class Demo4 {
    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);                
                }
                System.out.println("线程中止");
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }).start();
        // 3秒之后,将状态标志改为False,代表不继续裕兴
        Thread.sleep(3000L);
        flag = false;
        System.out.println("程序运行结束");
    }
}

  【执行结果】主线程休眠三秒才切换标志位值,所以匿名线程中的while循环体可以被执行三次,输出三次执行中,三秒过后,主线程修改标志位值,匿名线程判断到标志位为false,跳出循环体,线程执行结束中止。

运行中
运行中
运行中
程序运行结束
线程中止


四、总结

  • stop()会清除监视器锁信息,用它来中止线程会导致程序拦腰截断,带来线程安全问题,不建议使用
  • interrupt()让处于等待阻塞中的目标线程被中断,根据所处情况的不同会抛出中断异常或返回异常值,除此之外的情况会设置线程的中断状态,不会导致线程安全问题,是一种正确主流的中止线程的方式
  • 标志位适用于简单的程序,为了保证线程可见性通常要加上volatile修饰
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342