java concurrency: daemon线程
daemon线程的概念
在学习操作系统概念的时候,我们就曾听说过daemon的概念。daemon本身指的是在后台运行的进程或者线程,一般用来提供某些不需要与用户直接交互的服务,有点像我们见到的一些系统服务。在java线程中,一般可以分为两类,一类是普通的线程,就是那些我们通过常用的Thread类或者Runnable接口实现并启动的类。还有一类是daemon线程。这种线程也通过和创建普通线程同样的方式来创建,不过需要通过setDaemon方法设置为daemon线程。
daemon线程有几个典型的特征:
1. daemon线程是运行于后台的,不和用户直接交互。
2. 相对普通的线程来说,daeomn线程可以说是普通线程的下属,它的优先级要低一些。
3. daemon线程具有一定程度的继承性,这个继承性不是指类的继承,而是指当一个daemon线程创建一个新线程的话,这个新创建的线程就默认为daemon线程了。
4. 和普通线程的退出机制不同,在jvm即将关闭的时候,普通线程需要执行一系列的记录或者退出方法,比如需要执行finalize方法的则需要执行这部分,有finally块的代码也必须执行。而daemon线程退出的时候,jvm直接将它丢弃并退出,里面就算有finally块定义的代码也不会被执行。
一个daemon线程的示例
创建或者使用daemon线程一般包括一下3个步骤:
1. 如果线程本身就是daemon线程了,那么通过常规线程创建手段创建出来的就已经是daemon线程。
2. 如果通过普通线程创建创建出来的话,需要在启动线程前调用setDaemon()方法。切记,一定要在调用start()方法前。
3.我们可以通过调用isDaemon()方法来判断一个线程是否为daeomn.
下面是示例代码:
- class MyDaemon implements Runnable
- {
- Thread thrd;
- MyDaemon()
- {
- // Create the thread
- thrd = new Thread(this);
- // Set to daemon
- thrd.setDaemon(true);
- // Start the thread
- thrd.start();
- }
- public void run()
- {
- try
- {
- for(;;)
- {
- System.out.print(".");
- Thread.sleep(1000);
- }
- }
- catch(InterruptedException exc)
- {
- System.out.println("MyDaemon interrupted.");
- }
- }
- }
- class DaemonDemo
- {
- public static void main(String[] args)
- {
- MyDaemon dt = new MyDaemon();
- if(dt.thrd.isDaemon())
- System.out.println("dt is a daemon thread.");
- System.out.println("Sleeping in main thread.");
- try
- {
- Thread.sleep(10000);
- }
- catch(InterruptedException exc)
- {
- System.out.println("Main thread interrupted.");
- }
- System.out.println("\nMain thread ending.");
- }
- }
在上面的代码中,我们实现Runnable接口定义一个MyDaemon类,在构造函数中调用setDaemon()设置该线程为daemon.在main方法中启动MyDaemon线程之后,main方法结束时daemon线程也自动结束了。上述代码的执行结果如下:
- dt is a daemon thread.
- Sleeping in main thread.
- ..........
- Main thread ending.
一个比较有意思的事情就是,如果我们将前面代码里setDaemon()方法的这一行注释了,也就是说将该线程设置为普通的用户线程,再执行上面的代码时,我们会发现,程序不会终止,会一直执行下去,虽然主线程已经结束了。其输出的结果会像如下所示:
- Sleeping in main thread.
- ..........
- Main thread ending.
- .......................
设置daemon线程的意义
从前面的理解来看,daemon线程有点像一个我们画的草图,随时可以丢弃。它一般是用于在jvm关闭的时候,我们需要启动一些线程做一些辅助性的工作,但是我们又不希望这种工作妨碍到jvm正常关闭。和其他正常的线程来说,或许我们应该称之为不碍事的线程更合适。
daemon线程的一些应用
daemon线程在java中有很多地方用到,一个典型的就是垃圾回收的GC线程。另外如果我们需要用线程做一些比如时间提醒,标记等事情,采用daemon线程也是一个比较合适的选择。
又一个daemon的示例
根据前面的理解,我们再尝试一个更复杂的daemon应用。假定我们在一个应用中需要定义一些时间通知,比如说设置在某个指定的时候弹出一个会议通知的提醒。这是一个比较常见的场景,可以用daemon线程来实现。
具体的代码实现如下:
- import java.util.*;
- class Reminder implements Runnable
- {
- Calendar reminderTime;
- String message;
- Reminder(String msg, int delay)
- {
- message = msg;
- // Get the current time and date.
- reminderTime = Calendar.getInstance();
- // Add the delay to the time and date.
- reminderTime.add(Calendar.SECOND, delay);
- System.out.printf("Reminder set for %tD %1$tr\n", reminderTime);
- // Create the reminder thread.
- Thread dThrd = new Thread(this);
- // Set to daemon
- dThrd.setDaemon(true);
- // Start execution.
- dThrd.start();
- }
- // Run the reminder.
- public void run()
- {
- try
- {
- for(;;)
- {
- // Get the current time and date.
- Calendar curTime = Calendar.getInstance();
- // See if it's time for the reminder.
- if(curTime.compareTo(reminderTime) >= 0)
- {
- System.out.println("\n" + message + "\n");
- break;
- }
- Thread.sleep(1000);
- }
- }
- catch(InterruptedException exc)
- {
- System.out.println("Reminder interrupted.");
- }
- }
- }
- class ReminderDemo
- {
- public static void main(String[] args)
- {
- // Get a reminder 2 seconds from now.
- Reminder mt = new Reminder("Call Harry", 2);
- // Keep the main thread alive for 20 seconds.
- for(int i = 0; i < 20; i++)
- {
- try
- {
- Thread.sleep(1000);
- }
- catch(InterruptedException exc)
- {
- System.out.println("Main thread interrupted.");
- }
- System.out.print(".");
- }
- System.out.println("\nMain thread ending.");
- }
- }
我们设置了一个daemon线程,在一个无限循环里比较当前时间和指定的时间,在达到指定的时间时则显示一个提醒消息。上述代码的输出如下:
- Reminder set for 10/21/12 01:57:10 PM
- ..
- Call Harry
- ..................
- Main thread ending.
总结
悄悄的,daemon线程走了,正如daemon线程悄悄的来;挥一挥手,带不走jvm的一片云彩:)
java concurrency: daemon线程的更多相关文章
- 【转】关于Java的Daemon线程的理解
原文地址:http://www.cnblogs.com/ChrisWang/archive/2009/11/28/1612815.html 关于Java的Daemon线程的理解 网上对Java的Dae ...
- 深入浅出 Java Concurrency (35): 线程池 part 8 线程池的实现及原理 (3)[转]
线程池任务执行结果 这一节来探讨下线程池中任务执行的结果以及如何阻塞线程.取消任务等等. 1 package info.imxylz.study.concurrency.future;2 3 publ ...
- 深入浅出 Java Concurrency (34): 线程池 part 7 线程池的实现及原理 (2)[转]
线程池任务执行流程 我们从一个API开始接触Executor是如何处理任务队列的. java.util.concurrent.Executor.execute(Runnable) Executes t ...
- 深入浅出 Java Concurrency (33): 线程池 part 6 线程池的实现及原理 (1)[转]
线程池数据结构与线程构造方法 由于已经看到了ThreadPoolExecutor的源码,因此很容易就看到了ThreadPoolExecutor线程池的数据结构.图1描述了这种数据结构. 图1 Thre ...
- 深入浅出 Java Concurrency (30): 线程池 part 3 Executor 生命周期[转]
我们知道线程是有多种执行状态的,同样管理线程的线程池也有多种状态.JVM会在所有线程(非后台daemon线程)全部终止后才退出,为了节省资源和有效释放资源关闭一个线程池就显得很重要.有时候无法正确的关 ...
- 深入浅出 Java Concurrency (28): 线程池 part 1 简介[转]
从这一节开始正式进入线程池的部分.其实整个体系已经拖了很长的时间,因此后面的章节会加快速度,甚至只是一个半成品或者简单化,以后有时间的慢慢补充.完善. 其实线程池是并发包里面很重要的一部分,在实际情况 ...
- 深入浅出 Java Concurrency (36): 线程池 part 9 并发操作异常体系[转]
并发包引入的工具类很多方法都会抛出一定的异常,这些异常描述了任务在线程池中执行时发生的例外情况,而通常这些例外需要应用程序进行捕捉和处理. 例如在Future接口中有如下一个API: java.uti ...
- 深入浅出 Java Concurrency (29): 线程池 part 2 Executor 以及Executors[转]
Java里面线程池的顶级接口是Executor,但是严格意义上讲Executor并不是一个线程池,而只是一个执行线程的工具.真正的线程池接口是ExecutorService. 下面这张图完整描述了线程 ...
- Java Concurrency - 取消线程执行器中的线程
When you work with an executor, you don't have to manage threads. You only implement the Runnable or ...
随机推荐
- android 背景透明度渐变动画
button.setVisibility(View.VISIBLE); // 背景透明度渐变动画 ObjectAnimator alpha = ObjectAnimator.ofFloat(butto ...
- 如何从 0 开始学 ruby on rails (漫步版)
如何从 0 开始学 ruby on rails (漫步版) ruby 是一门编程语言,ruby on rails 是 ruby 的一个 web 框架,简称 rails. 有很多人对 rails 感兴 ...
- JS HTML DOM
HTML DOM (文档对象模型) 当网页被加载时,浏览器会创建页面的文档对象模型(Document Object Model). HTML DOM 模型被构造为对象的树. HTML DOM 树 通过 ...
- raphael入门到精通---入门篇之总览
什么是Raphael raphael.js是一小巧的javascript库,它可以在web上画矢量图简化你的工作,如果你想创建你指定的图表,图形区域或者可移动的组件,那么就使用raphael吧 话不多 ...
- JS 在html中的位置
前言 当我了解完html在浏览器中的解析渲染流程后,反而又发现了新的困扰自己的问题. Q:即然html要渲染需要渲染树,而渲染树又需要DOMTree和CSSRuleTree,DOMTree需要解析HT ...
- iOS中的地图和定位
文章摘自http://www.cnblogs.com/kenshincui/p/4125570.html#location 如有侵权,请联系删除. 概览 现在很多社交.电商.团购应用都引入了地图和定 ...
- final, finally, finalize 的区别
1.final 用于声明属性, 方法和类, 分别表示属性不可变, 方法不可覆盖, 类不可继承.内部类要访问局部变量, 局部变量必须定义成 final 类型, 例如, 一段代码…… 2.finally ...
- c++ primer plus 习题答案(3)
p296.3 #include<iostream> #include<cstdlib> #include<string> #include<cstring&g ...
- C++编程规范之23:头文件应该自给自足
摘要: 各司其责:应该确保所编写的每个头文件都能够独自进行编译,为此需要包含其内容所依赖的所有头文件. 如果一个文件包含某个头文件时,还要包含另一个头文件才能工作,就会增加交流障碍,给头文件的用户增添 ...
- sqlserver 只有函数和扩展存储过程才能从函数内部执行
一个SQLServer的自定义函数中调用一个自定义的存储过程,执行此函数后发出如下提示:“只有函数和扩展存储过程才能从函数内部执行". 原因:函数只能使用简单的sql语句,逻辑控制语句,复杂 ...