版权声明:本文出自汪磊的博客,转载请务必注明出处。

一、守护线程概述及示例

守护线程就是为其它线程提供"守护"作用,说白了就是为其它线程服务的,比如GC线程。

java程序中线程分两种:用户线程与守护线程,用户线程就是我们平常编写的一个个子线程,比如负责下载的线程,上传数据的线程等。如果一个线程调用了setDaemon(true)方法则变成了守护线程,两种线程本质上没什么区别,但是当一个工程中所有用户线程都执行完了,那么守护线程就没什么服务对象了,此时虚拟机退出,守护线程被销毁。

下面通过一个小Demo示例。编写DaemonThread类,如下:

 public class DaemonThread extends Thread {

     @Override
public void run() { while(true){
try {
Thread.sleep(1000);
System.out.println("我是线程"+Thread.currentThread().getName()+"我正在执行");
} catch (Exception e) {
//
e.printStackTrace();
}
}
}
}

很简单,在DaemonThread的run方法中先让线程休眠一秒钟,然后打印一下信息,接下来看下main中逻辑:

 public static void main(String[] args) {
//
DaemonThread d1 = new DaemonThread();
d1.setDaemon(true);
d1.start(); for(int i=0;i<500;i++){
System.out.println("我是线程"+Thread.currentThread().getName()+"我正在执行:"+i);
}
}

主要逻辑就是调用setDaemon方法将d1线程设置为守护线程,主线程中循环打印信息。运行打印如下信息:

 我是线程main我正在执行:0
我是线程main我正在执行:1
我是线程main我正在执行:2
我是线程main我正在执行:3
我是线程main我正在执行:4
我是线程main我正在执行:5
我是线程main我正在执行:6
我是线程main我正在执行:7
我是线程main我正在执行:8
我是线程main我正在执行:9
我是线程main我正在执行:10
   ....
我是线程main我正在执行:96
我是线程main我正在执行:97
我是线程main我正在执行:98
我是线程main我正在执行:99

我们看到只有主线程中信息打印,d1守护线程没有任何信息打印出,原因也很好解释了,运行程序主线程瞬间执行完毕,此时项目中没有其余线程工作,JVM也就退出了,进而d1线程也就得不到执行就被销毁了。守护线程介绍到此为止。

一、线程join()方法概述及示例

我所理解的join()的方法主要作用就是"等待"的作用,什么意思呢?比如B线程要等A线程执行完才开始执行其逻辑,那么就可以在B线程即将开始执行其逻辑的时候调用A线程的join()方法,此时就会转到A线程逻辑执行,执行完继续回来执行B线程逻辑,记住:要想join()方法起作用,A线程此时必须是alive状态(源码中有体现)。

直接看Demo吧.

JoinThread1类:很简单就是打印信息

 public class JoinThread1 extends Thread {

     @Override
public void run() { try {
int nextInt = new Random().nextInt(5);
Thread.sleep(nextInt * 1000);
System.out.println("我是线程"+Thread.currentThread().getName()+"我睡了"+nextInt+"秒");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

JoinThread2类:

 public class JoinThread2 extends Thread {

     private Thread mThread;

     public JoinThread2(Thread mThread) {
super();
this.mThread = mThread;
} @Override
public void run() { try {
mThread.join();
int nextInt = new Random().nextInt(5);
Thread.sleep(nextInt * 1000);//sleep在同步的方法中是不释放对象锁的,只有同步方法执行完毕,其他线程才可以执行。
System.out.println("我是线程"+Thread.currentThread().getName()+"我睡了"+nextInt+"秒"); } catch (InterruptedException e) {
e.printStackTrace();
}
}
}

初始化的时候传进来一个mThread,在run方法执行的时候首先调用传递进来的mThread的join()方法然后在执行其逻辑。

mian方法:

 public static void main(String[] args) {
//
try {
JoinThread1 join1 = new JoinThread1();
join1.start();
JoinThread2 join2 = new JoinThread2(join1);
join2.start();
join2.join();//join底层是wait方法,会释放对象锁的
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("我是主线程我执行完毕了");
}

main方法中逻辑也很简单,主要是初始化线程并启动,但是我们调用了join2.join()方法,所以主线程就要等待join2线程执行完毕才继续往下执行,运行程序输出如下:

 我是线程Thread-0我睡了1秒
我是线程Thread-1我睡了0秒
我是主线程我执行完毕了

如果你理解了上面说的打印信息顺序应该很容易理解。

但是如果main代码改为如下输出Log是什么样的呢:

public static void main(String[] args) {
//
try {
JoinThread1 join1 = new JoinThread1();
join1.start();
JoinThread2 join2 = new JoinThread2(join1);
join2.join();//join底层是wait方法,会释放对象锁的
join2.start();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("我是主线程我执行完毕了");
}

改为如下又是什么呢?

 public static void main(String[] args) {
//
try {
JoinThread1 join1 = new JoinThread1();
JoinThread2 join2 = new JoinThread2(join1);
join2.start();
join2.join();//join底层是wait方法,会释放对象锁的
join1.start();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("我是主线程我执行完毕了");
}

如果你能全部答对那么join()方法你就基本全部理解了,知识点虽小,但是也要认真理解透!!!

本文到此为止,希望对你有帮助。

java多线程之守护线程以及Join方法的更多相关文章

  1. Java多线程之守护线程

    Java多线程之守护线程 一.前言 Java线程有两类: 用户线程:运行在前台,执行具体的任务,程序的主线程,连接网络的子线程等都是用户线程 守护线程:运行在后台,为其他前台线程服务 特点:一旦所有用 ...

  2. Java 多线程基础(九)join() 方法

    Java 多线程基础(九)join 方法 一.join() 方法介绍 join() 定义 Thread 类中的,作用是:把指定的线程加入到当前线程,可以将两个交替执行的线程合并为顺序执行的线程.如:线 ...

  3. 多线程---其他方法 停止线程、守护线程、join方法

    第三方停止线程: 原来是stop(),因为该方法有些问题,所以被interrupt()方法取代,它的用途跟机制是 当没有指定的方式让冻结的线程恢复到运行状态时,这时需要对冻结进行清除,强制让线程恢复到 ...

  4. java多线程之守护线程与非守护线程

    在java线程中有两种线程,一种是用户线程,其余一种是守护线程. 守护线程具有特殊的含义,比如gc线程.当最后一个非守护线程执行完后,守护线程随着jvm一同结束工作. java中的守护线程需要将Dae ...

  5. Java多线程(2):线程加入/join()

    线程加入 join()方法,等待其他线程终止.在当前线程(主线程)中调用另一个线程(子线程)的join()方法,则当前线程转入阻塞状态,直到另一个线程运行结束,当前线程再由阻塞转为就绪状态. 也就是主 ...

  6. Java中的后台线程和join方法

    /*守护线程(后台线程):在一个进程中如果只剩下 了守护线程,那么守护线程也会死亡. 需求: 模拟QQ下载更新包. 一个线程默认都不是守护线程. */ public class Demo extend ...

  7. 43. 守护线程 和 join方法

    1.守护线程(后台线程):            我们在使用一款软件的时候,有的软件会让我们在不知道的情况下下载一些东西,那么这个就是后台线程.            一般用于提高软件的下载量(也就是 ...

  8. java多线程之守护线程(Daemon)

    https://blog.csdn.net/u010739551/article/details/51065923/

  9. 015.4守护线程和join

    内容:守护线程.join方法#####################守护线程通过开启线程之前调用setDaemon()方法,变成后台线程,前台线程运行完,后台线程自动会结束#########例子 c ...

随机推荐

  1. PMP杂谈--PMP中一些easy忽视的地方

    识别干系人:这个过程是持续的,在整个项目的生命周期中都要持续识别干系人. 组织过程资产和事业环境因素:这两个东西在过程的输入中似乎常常看到,但有时候又看不到,不要纠结了 ,不要浪费脑细胞去背诵哪个有, ...

  2. Gradle笔记——构建基础

    本篇文章主要介绍一下Gradle的构建基础. 它看起来似乎和android项目没多大关系.不能让人一下子理解android项目中的Gradle配置文件,可是这篇基础真的非常重要. 学习Gradle前, ...

  3. websocket简介

    在我们做web项目的过程中,经常需要做的一个模块是消息模块.典型的场景:一个商城系统的后台管理,我想实现如果前台有客户下单,后台就会接到消息,以便尽快发货处理. 要实现上述的功能,我们有几种备选的方案 ...

  4. Exception in thread "main" java.lang.NoClassDefFoundError: org/springframework/boot/context/embedded/ServletRegistrationBean

    异常信息 2017-09-02 18:06:37.223 [main] ERROR o.s.boot.SpringApplication - Application startup failed ja ...

  5. Postgres中的物化节点之sort节点

    顾名思义,物化节点是一类可缓存元组的节点.在执行过程中,很多扩展的物理操作符需要首先获取所有的元组后才能进行操作(例如聚集函数操作.没有索引辅助的排序等),这时要用物化节点将元组缓存起来.下面列出了P ...

  6. 《设计模式:可复用面向对象软件的基础》【PDF】下载

    <设计模式:可复用面向对象软件的基础>[PDF]下载链接: https://u253469.pipipan.com/fs/253469-230382288 内容介绍 <设计模式:可复 ...

  7. 字符串输入时的strlen()与\0

    在做串的数据结构时,被字符串printf %s输出时的结尾判断.strlen();长度.自定义StrCopy();字符串复制函数.StrAssign();字符串赋值函数卡了一下,固写此博巩固相关知识点 ...

  8. iOS NSString 文本不同的颜色 标题+文本字体大小 行间距/删除不需要的字符 /以及自适应高度

    #import <Foundation/Foundation.h> @interface TextsForRow : NSObject @property(nonatomic,copy)N ...

  9. html统计

    <!doctype html><html lang="en"> <head>  <meta charset="UTF-8&quo ...

  10. 新手了解.Nat

    1.Net平台 -->.Net平台 -->.Net FrameWrok框架 2.C#编程语言 -->编程语言:计算机能听懂的语言 -->使用.Net平台  C#是在.NEt平台 ...