深入理解Java中停止线程

一.停止线程会带来什么?

对于单线程中,停止单线程就是直接使用关键字return或者break,但是在停止多线程时是让线程在完成任务前去开启另外一条线程,必须放弃当前任务,而这个过程是不可预测,所以必须去做好防备。

二.认识停止线程的几个方法

 2.1三个被弃用的方法

 stop()、suspend()、resume()。

 stop()方法被弃用的原因:无论线程执行到了什么位置,一旦被stop就会被马上强制中断,并且释放线程所有持有锁对象,根本就没有安全性。

 suspend()和resume()这一对烂兄烂弟,因为只有其他线程调用resume这个方法时他才会释放suspend这个方法的锁,这样就极易造成死锁。

 2.2三个名字差不多的方法

 interrupt()中断线程、interrupted()判断当前线程是否停止、isInterrupted()判断线程是否停止。

 首先来介绍一下其中两个名字最相近的:interrupted()、isInterrupted()方法,这两个方法是用来测试线程是否被中断的。

 来看一下原码:

isInterrupted():

    public boolean isInterrupted() {

        return isInterrupted(false);

    }


 interrupted():

    public static boolean interrupted() {

        return currentThread().isInterrupted(true);

    }

可以明显看出interrupted()方法是静态的,而isInterrupted()是非静态的,但都是返回线程是否被中断。

下面我们来做一个测试,代码如下:

    public class Is_Interrupt extends Thread{

        @Override

        public void run(){

            for(int i=0;i<1000;i++){

                System.out.println("当前i的值为:"+i);

            }

        }

        public static void main(String[] args) {

            Is_Interrupt is_interrupt=new Is_Interrupt();

            is_interrupt.start();

            is_interrupt.interrupt();//中断线程

            System.out.println("线程是否已经暂停?"+is_interrupt.interrupted());

        }

    }

 输出结果为:

发现线程没有停止。

但是这里面还有一个线程就是main线程,而interrupted()返回的就是当前线程的中断状态,那么执行这个方法的就是main线程,而main线程此时当然没有中断。

我们将interrupted()方法改为isInterrupted()试试效果,代码如下:

    public class Is_Interrupt extends Thread{

        @Override

        public void run(){

            for(int i=0;i<1000;i++){

                System.out.println("当前i的值为:"+i);

            }

        }

        public static void main(String[] args) {

            Is_Interrupt is_interrupt=new Is_Interrupt();

            is_interrupt.start();

            is_interrupt.interrupt();//中断线程

            System.out.println("线程是否已经暂停?"+is_interrupt.isInterrupted());

        }

    }

结果如下:

说明线程是已经停止了的,只不过我们使用错了一个方法而已判断成了main线程的状态。

总结:isInterrupted方法是返回调用对象的中断状态,而静态方法interrupted是返回当前线程的中断状态。

既然了解了这个误区以后我们再来看看下面的代码:

public class InterruptText extends Thread{

    public static void main(String[] args) {

        System.out.println("main线程启动!");

        System.out.println( Thread.interrupted());//判断当前线程是否中断

        System.out.println(currentThread().isInterrupted());//通过currentThread().isInterrupted()同样也可以达到相同的目的,在单线程中

    }

}

输出:

这个是没有问题的。

那么我们多次调用这个interrupted方法呢?

public class InterruptText extends Thread{

    public static void main(String[] args) {

        System.out.println("main线程启动!");

        currentThread().interrupt();//中断main线程

        System.out.println( Thread.interrupted());//判断当前main线程是否中断

        System.out.println( Thread.interrupted());//再一次判断当main前线程是否中断

    }

}

 结果:

按照常理应该两次返回ture,但是为什么变成了第二次变成了false了呢?

其实interrupted就是在清除状态,你两次调用当然会将true变成flase但是他还是中断状态,但是isInterrupted是不清除的。

interrupt()方法:中断线程

三.停止线程

3.1通过异常来暂停线程

 首先来看一段代码:

public class InterruptText extends Thread{

    @Override

    public void run(){

        try {

            for (int i = 0; i < 100000; i++) {

                if (currentThread().isInterrupted()) {//如果线程中断

                    throw new InterruptedException();//抛出异常

                } else {

                    System.out.println(i);

                }

            }

            System.out.println("线程没有终止");

        }catch (InterruptedException e){

            e.printStackTrace();

        }

    }

    public static void main(String[] args) throws Exception {

        InterruptText interruptText=new InterruptText();

        interruptText.start();

        Thread.sleep(100);

        interruptText.interrupt();

    }

}

 结果:

异常停止方法的策略就是:在遇到中断时,抛出异常,扑捉异常。

3.2在睡眠中中断线程

 代码如下:

public class InterruptText2 extends Thread {

    @Override

    public void run(){

        try {

            Thread.sleep(10000);//线程睡眠

        } catch (InterruptedException e) {

            e.printStackTrace();

        }

    }

    public static void main(String[] args) {

        InterruptText2 interruptText2=new InterruptText2();

        interruptText2.start();//开启线程,但是线程处于睡眠状态

        interruptText2.interrupt();//在睡眠状态中断线程


    }

}

 结果为:

3.3线程让步

 方法:yield(),当前线程放弃所有的资源,去执行其他的任务。但是放弃资源的时间不可以预判的。

3.4守护线程

 守护线程的定义:守护线程是一种特殊的线程,区别于非守护线程,当程序中不存在非守护线程时,守护线程退出,程序退出。

设置守护线程:setDaemon(),参数为ture则该线程为守护线程。

在此我向大家推荐一个架构学习交流群。交流学习群号874811168 里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

出处:https://www.cnblogs.com/SAM-CJM/p/9787929.html

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

推荐阅读更多精彩内容