服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|JavaScript|易语言|

服务器之家 - 编程语言 - Java教程 - 详解Java线程-守护线程与用户线程

详解Java线程-守护线程与用户线程

2021-07-28 11:39情深见鹿 Java教程

这篇文章主要介绍了Java守护线程与用户线程,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

干java 开发这么多年, 之前一直没留意java 进程还区分守护进程和用户进程。守护进程这个概念最早还是在linux系统中接触的,直到近期使用java开发心跳检测功能时,使用timer时才发现原来java也有守护线程的概念。

1. java 线程

1.1 守护线程与用户线程

java 线程分为守护线程(daemonthread) 和 用户线程(userthread)两类.

  1. 通常情况下,我们使用thread 创建的线程在默认情况下都属于用户线程, 当在启动线程之前, 执行thread.setdaemon(true)时, 线程会变成守护线程。
  2. 其实在本质上,用户线程和守护线程并没有太大区别,唯一的区别就是会影响虚拟机的退出(程序的终止)。当jvm中只剩下守护线程时,虚拟机会退出,及程序终止;而当jvm中至少拥有一个用户线程时,jvm都不会退出。

1.2 守护线程特点

java 中的守护线程和linux 中的守护进程还是有些区别的, linux 守护进程时系统级别的, 当系统退出时, 才会终止. 而java 中的守护线程时jvm 级别的, 当jvm中无任何用户进程时, 守护进程销毁, jvm 退出, 程序终止. 笔者认为java 守护进程的最主要的特点有:

  1. 守护进程是运行在程序后台的线程
  2. 守护进程创建的进程,依然时守护进程
  3. 守护进程不会影响jvm的退出,当jvm只剩余守护进程时,jvm 进行退出
  4. 守护进行在jvm退出时,自动销毁

在开发java 守护线程时, 需要注意:

  1. 在线程启动之前, 及执行thread.start() 方法前, 设置thread.setdemon(true);
  2. 在守护线程中启动的子线程也是守护线程
  3. 守护线程不建议进行写操作, 因为守护进程随时可能结束。

1.3 守护线程适用场景

针对于守护线程的特点,笔者认为,java 守护线程通常可用于开发一些为其它用户线程服务的功能。比如说心跳检测,事件监听等。java 中最有名的守护进程当属gc(垃圾回收)

2. java 守护线程开发

开发守护进程时需要注意两点:

  1. 必须在线程启动之前,即执行thread.start()方法前, 执行thread.setdaemon(true) 方法,否则会抛出异常illegalthreadstateexception
  2. 守护线程中创建的线程也是守护线程

2.1 测试程序退出

  1. 注释thread.setdaemon(true)时, 线程为用户线程, 程序进行无线循环, 程序不终止。
  2. 设置thread.setdaemon(true)后, 线程变成守护线程, 程序直接终止, 仅输出一行信息"main thread done"。 因为程序执行完system 语句之后, main 程序作为唯一的一个用户线程执行结束了, jvm 中只剩下一个守护进程,所以jvm 便退出了。
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class testthread {
 
 public static void main(string[] args) {
 
  ansytask ansytask = new ansytask();
 
  thread thread = new thread(ansytask);
 
  // 设置线程为异步线程
  // thread.setdaemon(true);
 
  // 启动线程
  thread.start();
  
  system.out.println("main thread done");
 }
}
 
class ansytask implements runnable{
 
 @override
 public void run() {
  while (true){
   system.out.println(localdatetime.now() + "-hello,thread");
  }
 }
}

2.2 测试守护线程中创建新的线程

测试会发现, 默认情况下, 守护线程创建的子线程依然是守护线程,用户创建的守护线程依然时用户线程。也可以在创建子线程时通过setdaemon()方法修改.

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class testthread {
 
 public static void main(string[] args) throws interruptedexception {
 
  ansytask ansytask = new ansytask();
 
  thread thread = new thread(ansytask);
 
  // 设置线程为异步线程
  thread.setdaemon(true);
 
  // 启动线程
  thread.start();
 
  // 给守护线程点儿执行时间
  thread.sleep(1000l);
 }
}
 
class ansytask implements runnable{
 
 @override
 public void run() {
  thread thread = new thread("subthread");
  system.out.println(thread.getname() + " is daemon:" + thread.isdaemon());
 }
}

以上所述是小编给大家介绍的java守护线程与用户线程详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对服务器之家网站的支持!

延伸 · 阅读

精彩推荐