教你在停车前为线程配个“保镖” | 带你学《Java语言高级特性》之十三-阿里云开发者社区

开发者社区> Java开发者> 正文

教你在停车前为线程配个“保镖” | 带你学《Java语言高级特性》之十三

简介: 众所周知,死锁是一种十分危险的情况,所以暴力型的线程停止操作被禁用了。为保证线程的流畅运行,本节将带着读者优雅地停止线程,并在线程运行路上安排守护线程。

上一篇:一文速解生产者-消费者模式问题 | 带你学《Java语言高级特性》之十二
【本节目标】
通过阅读本节内容,你将学会借助循环实现对线程的较为安全的停止操作,并学会创建守护线程,保证相关线程的平稳运行。

优雅的停止线程

在多线操作之中如果要启动多线程使用的是Thread类中的start()方法,而如果对于多线程需要进行停止处理,Thread类原本提供有stop()方法,但是对于这些方法从JDK1.2版本就已经废除了,而且一直到现在也不再建议使用,而除了stop()之外还有几个方法也被禁用了:

  • 停止多线程:public final void stop();
  • 销毁多线程:public void destroy();
  • 挂起线程:public final void suspend()、暂停执行;
  • 恢复挂起的线程执行:public final void resume();

之所以废除掉这些方法,主要原因是因为这些方法可能导致线程死锁。所以,如果要想实现线程的停止需要通过一种柔和的方式来进行。
范例:实现线程柔和的停止

public class ThreadDemo {
    public static boolean flag = true;
    public static void main(String[] args) throws Exception {
        new Thread(() -> {
            long num = 0;
            while (flag) {
                try {
                    Thread.sleep(50);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "正在运行、num = " + num++);
            }
        }, "执行线程").start();
        Thread.sleep(200);     //运行200毫秒
        flag = false;      //停止线程
    }
}

image.png
图一 停止线程

万一现在有其他线程去控制这个flag的内容,那么这个时候对于线程的停止也不是说停就立刻停止的,而是会在执行中判断flag的内容来完成。

守护线程

现在假设有一个人并且他有一个保镖,那么这个保镖一定是在这个人活着时候进行守护,如果这个人死了,保镖就没用了。所以在多线程中可以进行守护线程的定义,也就是说如果现在主线程的程序或者其他线程还在执行的时候,那么守护线程将一直存在,并且运行在后台状态。
在Thread类中提供有如下的守护线程的操作方法:

  • 设置为守护线程:public final void setDaemon(boolean on);
  • 判断是否为守护线程:public final boolean isDaemon();

范例:使用守护线程

public class ThreadDemo {
    public static void main(String[] args) throws Exception {
        Thread userThread = new Thread(() -> {
            for (int x = 0 ; x < 10 ; x++) {
                try {
                    Thread.sleep(100) ;
                } catch (InterruptedException e) {
                    e.printStackTrace() ;
                }
                System.out.println(Thread.currentThread().getName() + "正在运行、x = " + x) ;
            }
        }, "用户线程") ;         //完成核心业务
        Thread daemonThread = new Thread(() -> {
            for (int x = 0; x < Integer.MAX_VALUE ; x++) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException e) {
                    e.printStackTrace() ;
                }
                System.out.println(Thread.currentThread().getName() + "正在运行、x = " + x) ;
            }
        }, "守护线程") ;     //完成核心业务
        daemonThread.setDaemon(true) ;        //设置为守护线程
        userThread.start() ;
        daemonThread.start() ;
    }
}

image.png
图二 守护线程

可以发现所有的守护线程都是围绕在用户线程的周围,如果程序执行完毕了,守护线程也就消失了,在整个JVM中最大的守护线程就是GC线程。
程序执行中GC线程会一直存在,如果程序执行完毕,GC线程也将消失。

想学习更多的Java的课程吗?从小白到大神,从入门到精通,更多精彩不容错过!免费为您提供更多的学习资源。
本内容视频来源于阿里云大学

下一篇:另类的“同步”-volatile关键字 | 带你学《Java语言高级特性》之十四
更多Java面向对象编程文章查看此处

版权声明:本文中所有内容均属于阿里云开发者社区所有,任何媒体、网站或个人未经阿里云开发者社区协议授权不得转载、链接、转贴或以其他方式复制发布/发表。申请授权请邮件developerteam@list.alibaba-inc.com,已获得阿里云开发者社区协议授权的媒体、网站,在转载使用时必须注明"稿件来源:阿里云开发者社区,原文作者姓名",违者本社区将依法追究责任。 如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件至:developer2020@service.aliyun.com 进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容。

分享:
+ 订阅

Java开发者成长课堂

官方博客
官网链接