Timer和TimerTask

转载 2016年08月29日 14:28:22

目录结构:

  • Timer和TimerTask
  • 一个Timer调度的例子
  • 如何终止Timer线程
  • 关于cancle方式终止线程
  • 反复执行一个任务
  • schedule VS. scheduleAtFixedRate
  • 一些注意点

1. Timer和TimerTask

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

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

2. 一个Timer调度的例子

 1import java.util.Timer; 2import java.util.TimerTask; 3 4publicclass TestTimer { 5 6publicstaticvoid main(String args[]){ 7         System.out.println("About to schedule task."); 8new Reminder(3); 9         System.out.println("Task scheduled.");10    }1112publicstaticclass Reminder{13        Timer timer;1415public Reminder(int sec){16             timer = new Timer();17             timer.schedule(new TimerTask(){18publicvoid 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的实现代码如下:
1public Timer() {2this("Timer-" + serialNumber());3    }45public 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方法的源码:

1publicvoid cancel() {2synchronized(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方法:

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

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

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

看看这个内部变量thread:

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

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

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

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

 1privatevoid mainLoop() { 2while (true) { 3try { 4                TimerTask task; 5boolean taskFired; 6synchronized(queue) { 7// Wait for queue to become non-empty 8while (queue.isEmpty() && newTasksMayBeScheduled) 9                        queue.wait();10if (queue.isEmpty())11break; // 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类的线程安全的。

Timer和TimerTask与线程的关系

1. Timer是一个定时器,它可以根据指定的时间,指定的执行周期来执行固定的任务TimerTask,例子如下: Timer与线程的关系,在Timer源代码中可现如下代码: 1. ...
  • kouwoo
  • kouwoo
  • 2015年11月06日 10:25
  • 2630

Timer与TimerTask的真正原理&使用介绍

其实就Timer来讲就是一个调度器,而TimerTask呢只是一个实现了run方法的一个类,而具体的TimerTask需要由你自己来实现,例如这样: Timer timer = new Time...
  • xieyuooo
  • xieyuooo
  • 2013年02月24日 18:23
  • 32002

Java并发编程:Timer和TimerTask

下面内容转载自:   http://blog.csdn.net/xieyuooo/article/details/8607220   其实就Timer来讲就是一个调度器,而TimerTas...
  • samjustin1
  • samjustin1
  • 2016年08月18日 10:41
  • 496

传统定时器技术 Timer and TimerTask

本文为张孝祥java并发课程的学习笔记。 java.util.Timer定时器,实际上是个线程,定时调度所拥有的TimerTasks。 一个TimerTask实际上就是一个拥有run方法的类,需要定时...
  • dlf123321
  • dlf123321
  • 2015年01月15日 15:29
  • 905

Android Timer 用法以及更新UI时的技巧

Timers主要是用来在后台运行一些任务。可以把Timer设置为守护线程。当调用cancel时所有已经安排的任务都没会被取消。        Timer中的任务是依次执行的,如果一个任务花很长时...
  • dxpqxb
  • dxpqxb
  • 2013年03月11日 15:35
  • 2287

TimerTask()中调用TextView.setText报错原因(非UI线程中 不能访问UI组件)

TimerTask()中调用TextView.setText报错原因 今天写了个程序子模块大体意思是在定时器中产生数据赋给TextView显示,结果程序在调用TextView.setText()时便...
  • zhangjikuan
  • zhangjikuan
  • 2014年03月18日 21:45
  • 1145

Java的Timer和TimerTask怎么结束

java中自带的定时器有Timer和TimerTask,但是运行起来要结束任务用cancel方法可以结束该任务,此时Timer线程还在运行,程序并没有退出,那么怎么结束Timer呢? 示例代码如下p...
  • zhuhao717
  • zhuhao717
  • 2015年04月02日 19:52
  • 5128

Java—Timer和TimerTask详解(常用API)

以下内容根据 The JavaTM Tutorial 和相关API doc翻译整理,以供日后参考: 1.概览 Timer是一种定时器工具,用来在一个后台线程计划执行指定任务。它可以计划执行...
  • weiguishan
  • weiguishan
  • 2015年02月23日 12:32
  • 1670

线程配合及Timer TimerTask理解

1、 认识Thread和Runnable  Java中实现多线程有两种途径:继承Thread类或者实现Runnable接口。Runnable是接口,建议用接口的方式生成线程,因为接口可以实现多继承,...
  • u010069940
  • u010069940
  • 2016年08月16日 17:26
  • 1127

Android中定时器Timer和TimerTask的启动,停止,暂停,继续等操作实例

下面是一个在Android中使用定时器Timer和TimerTask的启动,停止,暂停,继续等操作的demo。 需要注意的问题主要有两点: 1、Timer和TimerTask在调用cancel()...
  • dj0379
  • dj0379
  • 2016年03月13日 14:44
  • 21240
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:Timer和TimerTask
举报原因:
原因补充:

(最多只允许输入30个字)