JDK中的Timer和TimerTask详解

目录结构:

  • Timer和TimerTask

  • 一个Timer调度的例子

  • 如何终止Timer线程

  • 关于cancle方式终止线程

  • 反复执行一个任务

  • schedule
    VS. scheduleAtFixedRate

  • 一些注意点

1. Timer和TimerTask

  Timer是jdk中提供的一个定时器工具,使用的时候会在主线程之外起一个单独的线程执行指定的计划任务,可以指定执行一次或者反复执行多次。

  TimerTask是一个实现了Runnable接口的抽象类,代表一个可以被Timer执行的任务。

2. 一个Timer调度的例子


 1 import java.util.Timer;
2 import java.util.TimerTask;
3
4 public class TestTimer {
5
6 public static void main(String args[]){
7 System.out.println("About to schedule task.");
8 new Reminder(3);
9 System.out.println("Task scheduled.");
10 }
11
12 public static class Reminder{
13 Timer timer;
14
15 public Reminder(int sec){
16 timer = new Timer();
17 timer.schedule(new TimerTask(){
18 public void run(){
19 System.out.println("Time‘s up!");
20 timer.cancel();
21 }
22 }, sec*1000);
23 }
24 }
25 }

运行之后,在console会首先看到:

About to schedule task.
Task scheduled.

然后3秒钟后,看到

Time‘s up!

从这个例子可以看出一个典型的利用timer执行计划任务的过程如下:

  • new一个TimerTask的子类,重写run方法来指定具体的任务,在这个例子里,我用匿名内部类的方式来实现了一个TimerTask的子类

  • new一个Timer类,Timer的构造函数里会起一个单独的线程来执行计划任务。jdk的实现代码如下:


1     public Timer() {
2 this("Timer-" + serialNumber());
3 }
4
5 public Timer(String name) {
6 thread.setName(name);
7 thread.start();
8 }

  • 调用相关调度方法执行计划。这个例子调用的是schedule方法。

  • 任务完成,结束线程。这个例子是调用cancel方法结束线程。

3. 如何终止Timer线程

  默认情况下,创建的timer线程会一直执行,主要有下面四种方式来终止timer线程:

  • 调用timer的cancle方法

  • 把timer线程设置成daemon线程,(new
    Timer(true)创建daemon线程),在jvm里,如果所有用户线程结束,那么守护线程也会被终止,不过这种方法一般不用。

  • 当所有任务执行结束后,删除对应timer对象的引用,线程也会被终止。

  • 调用System.exit方法终止程序

4. 关于cancle方式终止线程

这种方式终止timer线程,jdk的实现比较巧妙,稍微说一下。

首先看cancle方法的源码:


1     public void cancel() {
2 synchronized(queue) {
3 thread.newTasksMayBeScheduled = false;
4 queue.clear();
5 queue.notify(); // In case queue was already empty.
6 }
7 }

没有显式的线程stop方法,而是调用了queue的clear方法和queue的notify方法,clear是个自定义方法,notify是Objec自带的方法,很明显是去唤醒wait方法的。

再看clear方法:


1     void clear() {
2 // Null out task references to prevent memory leak
3 for (int i=1; i<=size; i++)
4 queue[i] = null;
5
6 size = 0;
7 }

clear方法很简单,就是去清空queue,queue是一个TimerTask的数组,然后把queue的size重置成0,变成empty.还是没有看到显式的停止线程方法,回到最开始new
Timer的时候,看看new Timer代码:


1     public Timer() {
2 this("Timer-" + serialNumber());
3 }
4
5 public Timer(String name) {
6 thread.setName(name);
7 thread.start();
8 }

看看这个内部变量thread:


1     /**
2 * The timer thread.
3 */
4 private TimerThread thread = new TimerThread(queue);

不是原生的Thread,是自定义的类TimerThread.这个类实现了Thread类,重写了run方法,如下:


 1     public void run() {
2 try {
3 mainLoop();
4 } finally {
5 // Someone killed this Thread, behave as if Timer cancelled
6 synchronized(queue) {
7 newTasksMayBeScheduled = false;
8 queue.clear(); // Eliminate obsolete references
9 }
10 }
11 }

最后是这个mainLoop方法,这方法比较长,截取开头一段:


 1     private void mainLoop() {
2 while (true) {
3 try {
4 TimerTask task;
5 boolean taskFired;
6 synchronized(queue) {
7 // Wait for queue to become non-empty
8 while (queue.isEmpty() && newTasksMayBeScheduled)
9 queue.wait();
10 if (queue.isEmpty())
11 break; // Queue is empty and will forever remain; die

可以看到wait方法,之前的notify就是通知到这个wait,然后clear方法在notify之前做了清空数组的操作,所以会break,线程执行结束,退出。

5. 反复执行一个任务

通过调用三个参数的schedule方法实现,最后一个参数是执行间隔,单位毫秒。

6. schedule VS. scheduleAtFixedRate

这两个方法都是任务调度方法,他们之间区别是,schedule会保证任务的间隔是按照定义的period参数严格执行的,如果某一次调度时间比较长,那么后面的时间会顺延,保证调度间隔都是period,而scheduleAtFixedRate是严格按照调度时间来的,如果某次调度时间太长了,那么会通过缩短间隔的方式保证下一次调度在预定时间执行。举个栗子:你每个3秒调度一次,那么正常就是0,3,6,9s这样的时间,如果第二次调度花了2s的时间,如果是schedule,就会变成0,3+2,8,11这样的时间,保证间隔,而scheduleAtFixedRate就会变成0,3+2,6,9,压缩间隔,保证调度时间。

7. 一些注意点

  • 每一个Timer仅对应唯一一个线程。

  • Timer不保证任务执行的十分精确。

  • Timer类的线程安全的。

JDK中的Timer和TimerTask详解,布布扣,bubuko.com

时间: 2024-10-12 19:28:37

JDK中的Timer和TimerTask详解的相关文章

JDK中的Timer和TimerTask详解尊昨综做宗椎

http://www.ebay.com/cln/rxp_frjl/2015-01-28/165233238011 http://www.ebay.com/cln/7f1_tzrj/2015-01-28/165058925014 http://www.ebay.com/cln/t3l_lpdv/2015-01-28/165064633012 http://www.ebay.com/cln/rxp_frjl/2015-01-28/165233248011 http://www.ebay.com/cl

Timer和TimerTask详解

1.概览 Timer是一种定时器工具,用来在一个后台线程计划执行指定任务.它可以计划执行一个任务一次或反复多次.TimerTask一个抽象类,它的子类代表一个可以被Timer计划的任务. 简单的一个例程: import java.util.Timer;import java.util.TimerTask; /*** Simple demo that uses java.util.Timer to schedule a task to execute* once 5 seconds have pa

.NET中的Timer类型用法详解(转)

在.NET FrameWork中有多个Timer,那么怎么根据实际情况进行选择确实是一个问题. 总体而言,计时器共有以下四种: 多线程计时器: 1 System.Threading.Timer2 System.Timers.Timer 特殊环境的单线程计时器: 1 System.Windows.Forms.Timer(使用环境:Windows Forms Timer)2 System.Windows.Threading.DispatcherTimer( 使用环境:WPF timer); 单线程计

Linux下安装jdk报Permission denied以及chmod详解

一.发现问题 在Linux中安装jdk.bin的时候发现问题,报错./config.sh: line 103: /home/jdk.bin : Permission denied 修改权限:chmod 775 /home/jdk.bin 二.chmod命令详解 文件/目录权限设置命令:chmod 用于改变文件或目录的访问权限 用法1:其语法格式为:chmod [who] [opt] [mode] 文件/目录名        其中who表示对象,是以下字母中的一个或组合:u:表示文件所有者g:表示

oc中字典的实现方法详解

一:字典的基本概念 Foundation中的字典(NSDictionary,NSMutableDictionary)是由键-值对组成的数据集合.正如,我们在字典里查找单词的定义一样. 通过key(键),查找的对应的value(值),key通常是字符串对象,也可以是其他任意类型对象.在一个字典对象中,key的值必须是唯一的. 此外,字典对象的键和值不可以为空(nil),如果需要在字典中加入一个空值,可以加入NSNull对象 二:不可变字典-NSDictionary 1:初始化(以一个元素和多个元素

Android总结篇系列:Activity中几个主要函数详解

专注Android领域开发. 仰望星空,同时需要脚踏实地. ——好记性不如烂博客 Android总结篇系列:Activity中几个主要函数详解 Activity作为Android系统中四大基本组件之一,包含大量的与其他的各大组件.intent.widget以及系统各项服务等之间的交互的函数.在此,本文主要选取实际项目开发中常用的,但完全理解又需要有一定深入了解的几个函数进行讲解,后续本文会根据需要不断更新. 1. startActivityForResult / onActivityResult

【转载】lucene中Field.Index,Field.Store详解

lucene在doc.add(new Field("content",curArt.getContent(),Field.Store.NO,Field.Index.TOKENIZED)); Field有两个属性可选:存储和索引. 通过存储属性你可以控制是否对这个Field进行存储: 通过索引属性你可以控制是否对该Field进行索引. 事实上对这两个属性的正确组合很重要. Field.Index Field.Store 说明 TOKENIZED(分词) YES 被分词索引且存储 TOKE

【Unity编程】Unity中关于四元数的API详解

Unity中关于四元数的API详解 Quaternion类 Quaternion(四元数)用于计算Unity旋转.它们计算紧凑高效,不受万向节锁的困扰,并且可以很方便快速地进行球面插值. Unity内部使用四元数来表示所有的旋转. Quaternion是基于复数,并不容易直观地理解. 不过你几乎不需要访问或修改单个四元数参数(x,y,z,w); 大多数情况下,你只需要获取和使用现有的旋转(例如来自"Transform"),或者用四元数来构造新的旋转(例如,在两次旋转之间平滑插入). 大

Oracle中常用的to_Char用法详解

Oracle中常用的to_Char用法详解(有FMT的详细列表) The following are number examples for the to_char function. to_char(1210.73, '9999.9') would return '1210.7' to_char(1210.73, '9,999.99') would return '1,210.73' to_char(1210.73, '$9,999.00') would return '$1,210.73'