java类的基本组成和使用 中的Timer 类怎么使用

java当中的定时器的4种使用方式
投稿:shangke
字体:[ ] 类型:转载 时间:
这篇文章主要介绍了java当中的定时器的4种使用方式,有需要的朋友可以参考一下
对于开发游戏项目的同胞来说,Timer 这个东西肯定不会陌生,今天对以前自己经常使用的定时进行了一番小小的总结!没有写具体实现的原理,只是列举出了其中的四种比较常见的使用方法,相对而言,所以只要按照其所列举的例子仿照即可!
import java.util.C
import java.util.D
import java.util.T
import java.util.TimerT
public class TimeTest {
public static void main(String[] args) {
//timer2();
//timer3();
//timer4();
// 第一种方法:设定指定任务task在指定时间time执行 schedule(TimerTask task, Date time)
public static void timer1() {
Timer timer = new Timer();
timer.schedule(new TimerTask() {
public void run() {
System.out.println("-------设定要指定任务--------");
}, 2000);// 设定指定的时间time,此处为2000毫秒
// 第二种方法:设定指定任务task在指定延迟delay后进行固定延迟peroid的执行
// schedule(TimerTask task, long delay, long period)
public static void timer2() {
Timer timer = new Timer();
timer.schedule(new TimerTask() {
public void run() {
System.out.println("-------设定要指定任务--------");
// 第三种方法:设定指定任务task在指定延迟delay后进行固定频率peroid的执行。
// scheduleAtFixedRate(TimerTask task, long delay, long period)
public static void timer3() {
Timer timer = new Timer();
timer.scheduleAtFixedRate(new TimerTask() {
public void run() {
System.out.println("-------设定要指定任务--------");
// 第四种方法:安排指定的任务task在指定的时间firstTime开始进行重复的固定速率period执行.
// Timer.scheduleAtFixedRate(TimerTask task,Date firstTime,long period)
public static void timer4() {
Calendar calendar = Calendar.getInstance();
calendar.set(Calendar.HOUR_OF_DAY, 12); // 控制时
calendar.set(Calendar.MINUTE, 0);
calendar.set(Calendar.SECOND, 0);
Date time = calendar.getTime();
// 得出执行任务的时间,此处为今天的12:00:00
Timer timer = new Timer();
timer.scheduleAtFixedRate(new TimerTask() {
public void run() {
System.out.println("-------设定要指定任务--------");
}, time, 1000 * 60 * 60 * 24);// 这里设定将延时每天固定执行
您可能感兴趣的文章:
大家感兴趣的内容
12345678910
最近更新的内容
常用在线小工具java 定时器几种方式 -
- ITeye博客
博客分类:
spring-task
Spring定时任务的几种实现
近日项目开发中需要执行一些定时任务,比如需要在每天凌晨时候,分析一次前一天的日志信息,借此机会整理了一下定时任务的几种实现方式,由于项目采用spring框架,所以我都将结合
spring框架来介绍。
从实现的技术上来分类,目前主要有三种技术(或者说有三种产品):
Java自带的java.util.Timer类,这个类允许你调度一个java.util.TimerTask任务。使用这种方式可以让你的程序按照某一个频度执行,但不能在指定时间运行。一般用的较少,这篇文章将不做详细介绍。
使用Quartz,这是一个功能比较强大的的调度器,可以让你的程序在指定时间执行,也可以按照某一个频度执行,配置起来稍显复杂,稍后会详细介绍。
Spring3.0以后自带的task,可以将它看成一个轻量级的Quartz,而且使用起来比Quartz简单许多,稍后会介绍。
从作业类的继承方式来讲,可以分为两类:
作业类需要继承自特定的作业类基类,如Quartz中需要继承自org.springframework.scheduling.quartz.QuartzJobBean;java.util.Timer中需要继承自java.util.TimerTask。
作业类即普通的java类,不需要继承自任何基类。
注:个人推荐使用第二种方式,因为这样所以的类都是普通类,不需要事先区别对待。
从任务调度的触发时机来分,这里主要是针对作业使用的触发器,主要有以下两种:
每隔指定时间则触发一次,在Quartz中对应的触发器为:org.springframework.scheduling.quartz.SimpleTriggerBean
每到指定时间则触发一次,在Quartz中对应的调度器为:org.springframework.scheduling.quartz.CronTriggerBean
注:并非每种任务都可以使用这两种触发器,如java.util.TimerTask任务就只能使用第一种。Quartz和spring task都可以支持这两种触发条件。
二.用法说明
详细介绍每种任务调度工具的使用方式,包括Quartz和spring task两种。
第一种,作业类继承自特定的基类:org.springframework.scheduling.quartz.QuartzJobBean。
第一步:定义作业类
import org.quartz.JobExecutionC
import org.quartz.JobExecutionE
import org.springframework.scheduling.quartz.QuartzJobB
public class Job1 extends QuartzJobBean {
private static int i = 0;
//调度工厂实例化后,经过timeout时间开始执行调度
public void setTimeout(int timeout) {
this.timeout =
* 要调度的具体任务
protected void executeInternal(JobExecutionContext context)
throws JobExecutionException {
System.out.println("定时任务执行中…");
第二步:spring配置文件中配置作业类JobDetailBean
&bean name="job1" class="org.springframework.scheduling.quartz.JobDetailBean"&
&property name="jobClass" value="com.gy.Job1" /&
&property name="jobDataAsMap"&
&entry key="timeout" value="0" /&
&/property&
说明:org.springframework.scheduling.quartz.JobDetailBean有两个属性,jobClass属性即我们在java代码中定义的任务类,jobDataAsMap属性即该任务类中需要注入的属性值。
第三步:配置作业调度的触发方式(触发器)
Quartz的作业触发器有两种,分别是
org.springframework.scheduling.quartz.SimpleTriggerBean
org.springframework.scheduling.quartz.CronTriggerBean
第一种SimpleTriggerBean,只支持按照一定频度调用任务,如每隔30分钟运行一次。
配置方式如下:
&bean id="simpleTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean"&
&property name="jobDetail" ref="job1" /&
&property name="startDelay" value="0" /&&!-- 调度工厂实例化后,经过0秒开始执行调度 --&
&property name="repeatInterval" value="2000" /&&!-- 每2秒调度一次 --&
第二种CronTriggerBean,支持到指定时间运行一次,如每天12:00运行一次等。
配置方式如下:
&bean id="cronTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean"&
&property name="jobDetail" ref="job1" /&
&!—每天12:00运行一次 --&
&property name="cronExpression" value="0 0 12 * * ?" /&
关于cronExpression表达式的语法参见附录。
第四步:配置调度工厂
&bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"&
&property name="triggers"&
&ref bean="cronTrigger" /&
&/property&
说明:该参数指定的就是之前配置的触发器的名字。
第五步:启动你的应用即可,即将工程部署至tomcat或其他容器。
第二种,作业类不继承特定基类。
Spring能够支持这种方式,归功于两个类:
org.springframework.scheduling.timer.MethodInvokingTimerTaskFactoryBean
org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean
这两个类分别对应spring支持的两种实现任务调度的方式,即前文提到到java自带的timer task方式和Quartz方式。这里我只写MethodInvokingJobDetailFactoryBean的用法,使用该类的好处是,我们的任务类不再需要继承自任何类,而是普通的pojo。
第一步:编写任务类
public class Job2 {
public void doJob2() {
System.out.println("不继承QuartzJobBean方式-调度进行中...");
可以看出,这就是一个普通的类,并且有一个方法。
第二步:配置作业类
&bean id="job2"
class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean"&
&property name="targetObject"&
&bean class="com.gy.Job2" /&
&/property&
&property name="targetMethod" value="doJob2" /&
&property name="concurrent" value="false" /&&!-- 作业不并发调度 --&
说明:这一步是关键步骤,声明一个MethodInvokingJobDetailFactoryBean,有两个关键属性:targetObject指定任务类,targetMethod指定运行的方法。往下的步骤就与方法一相同了,为了完整,同样贴出。
第三步:配置作业调度的触发方式(触发器)
Quartz的作业触发器有两种,分别是
org.springframework.scheduling.quartz.SimpleTriggerBean
org.springframework.scheduling.quartz.CronTriggerBean
第一种SimpleTriggerBean,只支持按照一定频度调用任务,如每隔30分钟运行一次。
配置方式如下:
&bean id="simpleTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean"&
&property name="jobDetail" ref="job2" /&
&property name="startDelay" value="0" /&&!-- 调度工厂实例化后,经过0秒开始执行调度 --&
&property name="repeatInterval" value="2000" /&&!-- 每2秒调度一次 --&
第二种CronTriggerBean,支持到指定时间运行一次,如每天12:00运行一次等。
配置方式如下:
&bean id="cronTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean"&
&property name="jobDetail" ref="job2" /&
&!—每天12:00运行一次 --&
&property name="cronExpression" value="0 0 12 * * ?" /&
以上两种调度方式根据实际情况,任选一种即可。
第四步:配置调度工厂
&bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean"&
&property name="triggers"&
&ref bean="cronTrigger" /&
&/property&
说明:该参数指定的就是之前配置的触发器的名字。
第五步:启动你的应用即可,即将工程部署至tomcat或其他容器。
到此,spring中Quartz的基本配置就介绍完了,当然了,使用之前,要导入相应的spring的包与Quartz的包,这些就不消多说了。
其实可以看出Quartz的配置看上去还是挺复杂的,没有办法,因为Quartz其实是个重量级的工具,如果我们只是想简单的执行几个简单的定时任务,有没有更简单的工具,有!
请看我第下文Spring task的介绍。
Spring-Task
上节介绍了在Spring 中使用Quartz,本文介绍Spring3.0以后自主开发的定时任务工具,spring task,可以将它比作一个轻量级的Quartz,而且使用起来很简单,除spring相关的包外不需要额外的包,而且支持注解和配置文件两种
形式,下面将分别介绍这两种方式。
第一种:配置文件方式
第一步:编写作业类
即普通的pojo,如下:
import org.springframework.stereotype.S
public class TaskJob {
public void job1() {
System.out.println(“任务进行中。。。”);
第二步:在spring配置文件头中添加命名空间及描述
&beans xmlns="http://www.springframework.org/schema/beans"
xmlns:task="http://www.springframework.org/schema/task"
。。。。。。
xsi:schemaLocation="http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd"&
第三步:spring配置文件中设置具体的任务
&task:scheduled-tasks&
&task:scheduled ref="taskJob" method="job1" cron="0 * * * * ?"/&
&/task:scheduled-tasks&
&context:component-scan base-package=" com.gy.mytask " /&
说明:ref参数指定的即任务类,method指定的即需要运行的方法,cron及cronExpression表达式,具体写法这里不介绍了,详情见上篇文章附录。
&context:component-scan base-package="com.gy.mytask" /&这个配置不消多说了,spring扫描注解用的。
到这里配置就完成了,是不是很简单。
第二种:使用注解形式
也许我们不想每写一个任务类还要在xml文件中配置下,我们可以使用注解@Scheduled,我们看看源文件中该注解的定义:
@Target({java.lang.annotation.ElementType.METHOD, java.lang.annotation.ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Scheduled
public abstract String cron();
public abstract long fixedDelay();
public abstract long fixedRate();
可以看出该注解有三个方法或者叫参数,分别表示的意思是:
cron:指定cron表达式
fixedDelay:官方文档解释:An interval-based trigger where the interval is measured from the completion time of the previous task. The time unit value is measured in milliseconds.即表示从上一个任务完成开始到下一个任务开始的间隔,单位是毫秒。
fixedRate:官方文档解释:An interval-based trigger where the interval is measured from the start time of the previous task. The time unit value is measured in milliseconds.即从上一个任务开始到下一个任务开始的间隔,单位是毫秒。
下面我来配置一下。
第一步:编写pojo
import org.springframework.scheduling.annotation.S
import org.
@Component(“taskJob”)
public class TaskJob {
@Scheduled(cron = "0 0 3 * * ?")
public void job1() {
System.out.println(“任务进行中。。。”);
第二步:添加task相关的配置:
&?xml version="1.0" encoding="UTF-8"?&
&beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:task="http://www.springframework.org/schema/task"
xsi:schemaLocation="
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/jdbc/spring-jdbc-3.0.xsd
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd
http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-3.0.xsd"
default-lazy-init="false"&
&context:annotation-config /&
&!—spring扫描注解的配置
&context:component-scan base-package="com.gy.mytask" /&
&!—开启这个配置,spring才能识别@Scheduled注解
&task:annotation-driven scheduler="qbScheduler" mode="proxy"/&
&task:scheduler id="qbScheduler" pool-size="10"/&
说明:理论上只需要加上&task:annotation-driven /&这句配置就可以了,这些参数都不是必须的。
Ok配置完毕,当然spring task还有很多参数,我就不一一解释了,具体参考xsd文档http://www.springframework.org/schema/task/spring-task-3.0.xsd。
cronExpression的配置说明,具体使用以及参数请百度google
允许的特殊字符
, - * ? / L W C
1-12 或者 JAN-DEC
1-7 或者 SUN-SAT
, - * ? / L C #
年(可选)
? 你不想设置那个字段
下面只例出几个式子
CRON表达式
"0 0 12 * * ?"
每天中午十二点触发
"0 15 10 ? * *"
每天早上10:15触发
"0 15 10 * * ?"
每天早上10:15触发
"0 15 10 * * ? *"
每天早上10:15触发
"0 15 10 * * ? 2005"
2005年的每天早上10:15触发
"0 * 14 * * ?"
每天从下午2点开始到2点59分每分钟一次触发
"0 0/5 14 * * ?"
每天从下午2点开始到2:55分结束每5分钟一次触发
"0 0/5 14,18 * * ?"
每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发
"0 0-5 14 * * ?"
每天14:00至14:05每分钟一次触发
"0 10,44 14 ? 3 WED"
三月的每周三的14:10和14:44触发
"0 15 10 ? * MON-FRI"
每个周一、周二、周三、周四、周五的10:15触发
浏览 16086
lengchaotian
浏览: 201424 次
来自: 西安
czb6788782 写道亲,你这Global类能给一份么
[size=small][/size]u;8i
定时任务在启动的时候可以传参数么?
亲,你这Global类能给一份么java定时器的使用(Timer)_文档下载
亿万文档 免费下载
当前位置: &
& java定时器的使用(Timer)
java定时器的使用(Timer)
在应用开发中,经常需要一些周期性的操作,比如每5分钟执行某一操作等。
对于这样的操作最方便、高效的实现方式就是使用java.util.Timer工具类。
java定时器的使用(Timer)1、在应用开发中,经常需要一些周期性的操作,比如每5分钟执行某一操作等。对于这样的操作最方便、高效的实现方式就是使用java.util.Timer工具类。private java.util.Ttimer = new Timer(true);timer.schedule(new java.util.TimerTask() { public void run(){ //server.checkNewMail(); 要操作的方法} }, 0, 5*60*1000);第一个参数是要操作的方法,第二个参数是要设定延迟的时间,第三个参数是周期的设定,每隔多长时间执行该操作。使用这几行代码之后,Timer本身会每隔5分钟调用一遍server.checkNewMail()方法,不需要自己启动线程。Timer本身也是多线程同步的,多个线程可以共用一个Timer,不需要外部的同步代码。2、(1)Timer.schedule(TimerTask task,Date time)安排在制定的时间执行指定的任务。(2)Timer.schedule(TimerTask task,Date firstTime ,long period)安排指定的任务在指定的时间开始进行重复的固定延迟执行.(3)Timer.schedule(TimerTask task,long delay)安排在指定延迟后执行指定的任务.(4)Timer.schedule(TimerTask task,long delay,long period)安排指定的任务从指定的延迟后开始进行重复的固定延迟执行.(5)Timer.scheduleAtFixedRate(TimerTask task,Date firstTime,long period)安排指定的任务在指定的时间开始进行重复的固定速率执行.(6)Timer.scheduleAtFixedRate(TimerTask task,long delay,long period)安排指定的任务在指定的延迟后开始进行重复的固定速率执行.用Java Timer API 进行时间调度开发的相关注意点java.util这个包中可以找到Timer和TimerTask这两个类。Timer直接从Object继承,它相当于一个计时器,能够用它来指定某个时间来执行一项任务,或者每隔一定时间间隔反复执行同一个任务。创建一个Timer后,就会生成一个线程在背后运行,来控制任务的执行。而TimerTask就是用来实现某项任务的类,它实现了Runnable接口,因此相当于一个线程。如何实现自己的任务调度?1、继承TimerTask,注意TimerTask是实现Runnable接口的,因此只要重载run()方法即可。2、创建Timer对象,调用schedule()方法。相关注意点分析:1、任务调度要优先考虑实时保证由于Java的天性,并且在开发JDK的过程中要考虑到不同平台,而不同平台的线程调度机制是不同的,因此各种平台下JVM 的线程调度机制也是不一致的。从而Timer不能保证任务在所指定的时间内执行。另外由于TimerTask是实现Runnable接口的,在TimerTask被放进线程队列睡眠一段时间(wait)之后,当到了指定的该唤起该TimerTask时,由于执行的确切时机取决于JVM的调度策略和当前还有
Word文档免费下载: (下载1-7页,共7页)
java定时器Timer的简单例子_计算机软件及应用_IT/计算机_专业资料。package com...{ timer.cancel();// 使用这个方法退出任务 } } catch (IOException e) { ...定时调用程序(java中的定时器Timer)_IT/计算机_专业资料。定时调用程序: 1.首先编写一个方法,定时器要定时执行的方法。比如: public class UpdateLucene { public...java Timer定时器_计算机软件及应用_IT/计算机_专业资料。java 定时器、Timerjava.util.Timer 定时器,实际上是个线程,定时调度所拥有的 TimerTasks。 一个 ...7. Timer.java:518:TimerThread.mainloop() 从上面的代码可以看出,如果执行的任务中有其他任何运行时异常 (RuntimeException)抛出,就必然导致定时器取消,也就是说...java定时器使用_IT/计算机_专业资料。java 定时器使用 . 分类: java 18:15230 人阅读评论(0)收藏举报 定时器类 Timer 在 java.util 包中。 ...java简单定时器timer_IT/计算机_专业资料。timer java定时器Java 简单定时器 主要是能运行 详细介绍网上多的是 可以自己去查找 1:在 web.xml 当中写入 在 &...java_Timer定时器实例_Spring实例_IT/计算机_专业资料。java_Timer定时器实例_... 下面是一个使用例子 TimerTest.java package com.ly. import java.io.*...java定时器详解(附详细代码)_IT/计算机_专业资料。java定时器详解(附详细代码)java 类 Timer 和 TimerTask 的使用这两个类使用起来非常方便,可以完成我们对定时...登录 注册 batitan
学习黑客技术 定时任务:Java中Timer和TimerTask的使用 Java Swing java.util.Timer定时器,实际...本文引用自Jason《java定时器的使用【转载】》 引用 Jason 的 java定时器的使用【转载】 java定时器的使用 定时器类Timer在java.util包中。使用时,先实例化,...在需要按时间计划执行简单任务的情况下,Timer是最常被使用到的工具类。使用Timer来调度TimerTask的实现者来执行任务,有两种方式,一种是使任务在指定时间被执行一次,另一种是从某一指定时间开始周期性地执行任务。
下面是一个简单的Timer例子,它每隔10秒钟执行一次特定操作doWork。
Timer timer = new Timer();
&&&&&&&&&&&&&&&&&&&&&&& TimerTask& task = new TimerTask (){
&&&&&&&&&&&&&&&&&&&&&&& public void run() {
&&&&&&&&&&&&&&&&&&&&&&&&&&&& doWork();
&&&&&&&&&&&&&&&&&&&&&&&&&& }
&&&&&&&&&&&&& timer.schedule (task, 10000L, 10000L);
可以看到,具体的任务由TimerTask的子类实现,Timer负责管理、执行TimerTask。
Timer 的使用
在不同的场景下,需要使用不同的Timer接口。如上所说,主要区分两种情况
1) 在指定时间执行任务,只执行一次
-&&&&&&&&& public void schedule(TimerTask task, long delay)
-&&&&&&&&& public void schedule(TimerTask task, Date time)
2)从指定时间开始,周期性地重复执行,直到任务被cancel掉。其中又分两种类型:
2.1) 一种是按上一次任务执行的时间为依据,计算本次执行时间,可以称为相对时间法。比如,如果第一次任务是1分10秒执行的,周期为5秒,因系统繁忙(比如垃 圾回收、虚拟内存切换),1分15秒没有得到机会执行,直到1分16秒才有机会执行第二次任务,那么第3次的执行时间将是1分21秒,偏移了1秒。
-&&&&&&&&& public void schedule(TimerTask task, long delay, long period)
-&&&&&&&&& public void schedule(TimerTask task, Date firstTime, long period)
2.2) 另一种是绝对时间法,以用户设计的起始时间为基准,第n次执行时间为“起始时间+n*周期时间”。比如,在上面的情况下,虽然因为系统繁忙,第二执行时间被推后1秒,但第3次的时间点仍然应该是1分20秒。
-&&&&&&&&& public void scheduleAtFixedRate(TimerTask task, long delay, long period)
-&&&&&&&&& public void scheduleAtFixedRate(TimerTask task, Date firstTime,&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& long period)
相 对时间法,关注于满足短时间内的执行间隔,绝对时间法,则更关注在一个长时间范围内,任务被执行的次数。如果我们要编写一个程序,用timer控制文档编 辑器中提示光标的闪烁,用哪种更合适? 当然是相对时间法。如果改用绝对时间法,当从系统繁忙状态恢复后,光标会快速连续闪烁多次,以弥补回在系统繁忙期间没有被执行的任务,这样的情况会用户来 说比较难以接受。又如,每10分钟检查一次新邮件的到来,也适合于使用相对时间法。
Timer timer = new Timer();
&&&&&&&&&&&&&&&&&&&&&&& TimerTask& task = new TimerTask (){
&&&&&&&&&&&&&&&&&&&&&&&& public void run() {
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& displayCursor();
&&&&&&&&&&&&&&&&&&&&&&&&&& }
&&&&&&&&&&&&& timer.schedule (task, 1000L, 1000L); //每秒闪烁一次光标
作为对比,我们来考虑一种绝对时间法的应用场景——倒数任务,比如,要求在10秒内做倒数计时,每秒做一次doworkPerSecond操作,10秒结束时做一次doworkEnd操作,然后结束任务。
Timer timer = new Timer();
&&&&&&&&&&&&&&&&&&&&&&& TimerTask& task = new TimerTask (){
&&&&&&&&&&&&&&&&&&&&&&&&&&&&& private int count=10;
&&&&&&&&&&&&&&&&&&&&&&&&&&&&& public void run() {
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& if(count&0){
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& doWorkPerSecond();
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& count--;
&&&&&&&&&&&&&&&&&&&&&&&&&&& }else{
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& doWorkEnd();
&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& cancel();
&&&&&&&&&&&&&&&&&&&&&&&&&&&& }
&&&&&&&&&&&&&&&&&&&&&&&&&& }
&&&&&&&&&&&&&&&&&&&&&&& timer. scheduleAtFixedRate (task, 1000L, 1000L);
Timer及相关类的内部实现
Timer的内部会启动一个线程TimerThread。即使有多个任务被加入这个Timer,它始终只有一个线程来管理这些任务。
-&&&&&&&&& TimerThread是Thread的子类。加入Timer的所有任务都会被最终放入TimerThread所管理的TaskQueue中。 TimerThread会不断查看TaskQueue中的任务,取出当前时刻应该被执行的任务执行之,并且会重新计算该任务的下一次执行时间,重新放入 TaskQueue。直到所有任务执行完毕(单次任务)或者被cancel(重复执行的任务),该线程才会结束。
-&&&&&&&&& TaskQueue,由数组实现的二叉堆,堆的排序是以任务的下一次执行时间为依据的。二叉堆的使用使得TimerThread以简洁高效的方式快速找到 当前时刻需要执行的TimerTask,因为,堆排序的特性是保证最小(或者最大)值位于堆叠顶端,在这里,queue[1]始终是下次执行时间 (nextExecutionTime)最小的,即应该最先被执行的任务
比如,同一个timer管理两个任务task1和task2
timer.schedule (task1, 4000L, 10000L);
timer. scheduleAtFixedRate (task2, 2000L, 15000L);
则,TaskQueue 中会有两个任务:task1和task2。task2会排在头部queue[1],当task2执行时间到,task2被执行,同时修改其 nextExecutionTime =当前的nextExecutionTime +15000L(绝对时间法)并重新在二叉堆中排序。排序后,task1被放到头部。当task1执行时间到,task1被执行,并修改其 nextExecutionTime =当前时间+10000L,然后重新在二叉堆中对其排序………
当收到客户端请求时,服务端生成一个Response对象。服务端希望客户端访问该对象的间隔时间不能超过20秒,否则,服务端认为客户端已经异常关闭或者网络异常,此时销毁掉该对象并打印错误日志。每次访问都会重新开始计时。
class Response{
private TimerTask&
&&&&&&&&&&& public void init(){
&&&&&&&& ………
Timer timer = new Timer();
timeout = new TimeOutTask();
&&&&&&&&&&&&&&&&&&&&&&& timer.schedule (timeout, 20000L);
public void invoke(){
&&& timeout.cancel();//取消当前的timeout任务
&& timeout = new TimeOutTask();
&&&&&&&&&&&&&&&&&&&&&&&&&& timer.schedule (timeout, 20000L);//重新开始计时
void destroy(){
class TimeOutTask extends TimerTask{
&&&&&&&&&& public void run() {
TraceTool.error(“Time out, destroy the Response object.”);
destroy();
&&&&&&&&&&&&&&&&&&&&&&&&&& }
因为Timer不支持对任务重置计时,所以此处采取了先cancel当前的任务再重新加入新任务来达到重置计时的目的。注意,对一个已经cancel的任务,不能通过schedule重新加入Timer中执行。TimerTask的状态机如下:
一个新生成的TimerTask其状态为VIRGIN,Timer只接受状态为VIRGIN的任务,否则会有IllegalStateException异常抛出。
调用任务的cancel方法,该任务就转入CANCELLED状态,并很快从TaskQueue中删除。对单次执行的任务,一旦执行结束,该任务也会从中删除。这意味着TimerTask将不再被timer所执行了。
定时器的使用(Timer)
1、在应用开发中,经常需要一些周期性的操作,比如每5分钟执行某一操作等。
对于这样的操作最方便、高效的实现方式就是使用java.util.Timer工具类。
private java.util.T
timer = new Timer(true);
timer.schedule(
new java.util.TimerTask() { public void run()
{ //server.checkNewMail(); 要操作的方法} }, 0, 5*60*1000);
第一个参数是要操作的方法,第二个参数是要设定延迟的时间,第三个参
数是周期的设定,每隔多长时间执行该操作。
使用这几行代码之后,Timer本身会每隔5分钟调用一遍
server.checkNewMail()方法,不需要自己启动线程。Timer本身也是多线程同
步的,多个线程可以共用一个Timer,不需要外部的同步代码。
(1)Timer.schedule(TimerTask task,Date time)安排在制定的时间执行指定的
(2)Timer.schedule(TimerTask task,Date firstTime ,long period)安排指定
的任务在指定的时间开始进行重复的固定延迟执行.
(3)Timer.schedule(TimerTask task,long delay)安排在指定延迟后执行指定的
(4)Timer.schedule(TimerTask task,long delay,long period)安排指定的任务
从指定的延迟后开始进行重复的固定延迟执行.
(5)Timer.scheduleAtFixedRate(TimerTask task,Date firstTime,long period)
安排指定的任务在指定的时间开始进行重复的固定速率执行.
(6)Timer.scheduleAtFixedRate(TimerTask task,long delay,long period)安
排指定的任务在指定的延迟后开始进行重复的固定速率执行.
用Java Timer API进行时间调度开发的相关注意点
java.util这个包中可以找到Timer和TimerTask这两个类。Timer直接从Object
继承,它相当于一个计时器,能够用它来指定某个时间来执行一项任务,或者
每隔一定时间间隔反复执行同一个任务。创建一个Timer后,就会生成一个线程
在背后运行,来控制任务的执行。而TimerTask就是用来实现某项任务的类,
它实现了Runnable接口,因此相当于一个线程。
如何实现自己的任务调度?
1、继承TimerTask,注意TimerTask是实现Runnable接口的,因此只要重载run()
方法即可。
2、创建Timer对象,调用schedule()方法。
相关注意点分析:
1、任务调度要优先考虑实时保证
由于Java的天性,并且在开发JDK的过程中要考虑到不同平台,而不同平台的
线程调度机制是不同的,因此各种平台下JVM的线程调度机制也是不一致的。
从而Timer不能保证任务在所指定的时间内执行。另外由于TimerTask是实现
Runnable接口的,在TimerTask被放进线程队列睡眠一段时间(wait)之后,
当到了指定的该唤起该TimerTask时,由于执行的确切时机取决于JVM的调度策
略和当前还有多少线程在等待CPU处理。因此就不能保证任务在所指定的时间
内执行。通常在如下两种情况下导致任务延迟执行:
(1)、有大量线程在等待执行
(2)、GC机制的影响导致延迟
这也是为什么在Timer API中存在两组调度方法的原因。即:
(1)、schedule()
用固定延迟调度。使用本方法时,在任务执行中的每一个延迟会传播到后续的任
务的执行。
(2)、scheduleAsFixedRate()
用固定比率调度。使用本方法时,所有后续执行根据初始执行的时间进行调度,
从而希望减小延迟。
具体使用哪一个方法取决于哪些参数对你的程序或系统更重要。
2、每个Timer对象要在后台启动一个线程。这种性质在一些托管的环境下不推
荐使用,比如在应用服务器中。因为这些线程不在容器的控制范围之内了。
具体Java API中的Timer类和TimerTask类的描述如下:
java.lang.Object
java.util.Timer
public class Timer
extends Object
一种线程设施,用于安排以后在后台线程中执行的任务。可安排任务执行一次,
或者定期重复执行。
与每个Timer对象相对应的是单个后台线程,用于顺序地执行所有计时器任务。
计时器任务应该迅速完成。如果完成某个计时器任务的时间太长,那么它会“独
占”计时器的任务执行线程。因此,这就可能延迟后续任务的执行,而这些任务
就可能“堆在一起”,并且在上述令人讨厌的任务最终完成时才能够被快速连续
对Timer对象最后的引用完成后,并且所有未处理的任务都已执行完成后,计
时器的任务执行线程会正常终止(并且成为垃圾回收的对象)。但是这可能要很
长时间后才发生。默认情况下,任务执行线程并不作为守护线程来运行,所以
它能够阻止应用程序终止。如果调用方想要快速终止计时器的任务执行线程,那
么调用方应该调用计时器的cancel方法。
如果意外终止了计时器的任务执行线程,例如调用了它的stop方法,那么所有
以后对该计时器安排任务的尝试都将导致IllegalStateException,就好像调用
了计时器的cancel方法一样。
此类是线程安全的:多个线程可以共享单个Timer对象而无需进行外部同步。
此类不提供实时保证:它使用Object.wait(long)方法来安排任务。
实现注意事项:此类可扩展到大量同时安排的任务(存在数千个都没有问题)。
在内部,它使用二进制堆来表示其任务队列,所以安排任务的开销是O(log n),
是同时安排的任务数。
实现注意事项:所有构造方法都启动计时器线程。
从以下版本开始:
另请参见:
TimerTask,Object.wait(long)
构造方法摘要
创建一个新计时器。
Timer(boolean isDaemon)
创建一个新计时器,可以指定其相关的线程作为守护程序运行。
Timer(Stringname)
创建一个新计时器,其相关的线程具有指定的名称。
Timer(Stringname, boolean isDaemon)
创建一个新计时器,其相关的线程具有指定的名称,并且可以指定作为守护程序运
void cancel()
终止此计时器,丢弃所有当前已安排的任务。
int purge()
从此计时器的任务队列中移除所有已取消的任务。
void schedule(TimerTasktask,Date
安排在指定的时间执行指定的任务。
void schedule(TimerTasktask,Date
firstTime, long period)
安排指定的任务在指定的时间开始进行重复的固定延迟执行。
void schedule(TimerTasktask, long delay)
安排在指定延迟后执行指定的任务。
void schedule(TimerTasktask, long delay, long period)
安排指定的任务从指定的延迟后开始进行重复的固定延迟执行。
void scheduleAtFixedRate(TimerTasktask,Date
firstTime, long period)
安排指定的任务在指定的时间开始进行重复的固定速率执行。
void scheduleAtFixedRate(TimerTasktask, long delay, long period)
安排指定的任务在指定的延迟后开始进行重复的固定速率执行。
从类java.lang.Object继承的方法
equals, finalize,getClass,hashCode,
notify,notifyAll,toString,
wait,wait,wait
构造方法详细信息
public Timer()
创建一个新计时器。相关的线程不作为守护程序运行。
另请参见:
Thread,cancel()
public Timer(boolean isDaemon)
创建一个新计时器,可以指定其相关的线程作为守护程序运行。如果计时器将用于
安排重复的“维护活动”,则调用守护线程,在应用程序运行期间必须调用守护线程,
但是该操作不应延长程序的生命周期。
isDaemon -如果应该将相关的线程作为守护程序运行,则为true。
另请参见:
Thread,cancel()
public Timer(Stringname)
创建一个新计时器,其相关的线程具有指定的名称。相关的线程不作为守护程序运
name -相关线程的名称。
NullPointerException -如果name为null。
从以下版本开始:
另请参见:
Thread.getName(),Thread.isDaemon()
public Timer(Stringname,
boolean isDaemon)
创建一个新计时器,其相关的线程具有指定的名称,并且可以指定作为守护程序运
name -相关线程的名称。
isDaemon -如果应该将相关的线程作为守护程序运行,则为true。
NullPointerException -如果name为null。
从以下版本开始:
另请参见:
Thread.getName(),Thread.isDaemon()
方法详细信息
public void schedule(TimerTasktask,
long delay)
安排在指定延迟后执行指定的任务。
task -所要安排的任务。
delay -执行任务前的延迟时间,单位是毫秒。
IllegalArgumentException -如果delay是负数, 或者delay +
System.currentTimeMillis() 是负数。
IllegalStateException -如果已经安排或取消了任务,或者已经取消计时器。
public void schedule(TimerTasktask,
Date time)
安排在指定的时间执行指定的任务。如果此时间已过去,则安排立即执行该任务。
task -所要安排的任务。
time -执行任务的时间。
IllegalArgumentException -如果time.getTime()是负数。
IllegalStateException -如果已经安排或取消了任务,已经取消了计时器,或者计时
器线程已终止。
public void schedule(TimerTasktask,
long delay,
long period)
安排指定的任务从指定的延迟后开始进行重复的固定延迟执行。以近似固定的时间
间隔(由指定的周期分隔)进行后续执行。
在固定延迟执行中,根据前一次执行的实际执行时间来安排每次执行。如
果由于任何原因(如垃圾回收或其他后台活动)而延迟了某次执行,则后
续执行也将被延迟。从长期来看,执行的频率一般要稍慢于指定周期的倒
数(假定Object.wait(long)所依靠的系统时钟是准确的)。
固定延迟执行适用于那些需要“平稳”运行的重复活动。换句话说,它适
用于在短期运行中保持频率准确要比在长期运行中更为重要的活动。这包
括大多数动画任务,如以固定时间间隔闪烁的光标。这还包括为响应人类
活动所执行的固定活动,如在按住键时自动重复输入字符。
task -所要安排的任务。
delay -执行任务前的延迟时间,单位是毫秒。
period -执行各后续任务之间的时间间隔,单位是毫秒。
IllegalArgumentException -如果delay是负数, 或者delay +
System.currentTimeMillis() 是负数。
IllegalStateException -如果已经安排或取消了任务,已经取消了计时器,或者计时
器线程已终止。
public void schedule(TimerTasktask,
Date firstTime,
long period)
安排指定的任务在指定的时间开始进行重复的固定延迟执行。以近似固定的时间间
隔(由指定的周期分隔)进行后续执行。
在固定延迟执行中,根据前一次执行的实际执行时间来安排每次执行。如
果由于任何原因(如垃圾回收或其他后台活动)而延迟了某次执行,则后
续执行也将被延迟。在长期运行中,执行的频率一般要稍慢于指定周期的
倒数(假定Object.wait(long)所依靠的系统时钟是准确的)。
固定延迟执行适用于那些需要“平稳”运行的重复执行活动。换句话说,
它适用于在短期运行中保持频率准确要比在长期运行中更为重要的活动。
这包括大多数动画任务,如以固定时间间隔闪烁的光标。这还包括为响应
人类活动所执行的固定活动,如在按住键时自动重复输入字符。
task -所要安排的任务。
firstTime -首次执行任务的时间。
period -执行各后续任务之间的时间间隔,单位是毫秒。
IllegalArgumentException -如果time.getTime()是负数。
IllegalStateException -如果已经安排或取消了任务,已经取消了计时器,或者计时
器线程已终止。
scheduleAtFixedRate
public void scheduleAtFixedRate(TimerTasktask,
long delay,
long period)
安排指定的任务在指定的延迟后开始进行重复的固定速率执行。以近似固定的时间
间隔(由指定的周期分隔)进行后续执行。
在固定速率执行中,根据已安排的初始执行时间来安排每次执行。如果由
于任何原因(如垃圾回收或其他背景活动)而延迟了某次执行,则将快速
连续地出现两次或更多的执行,从而使后续执行能够“追赶上来”。从长
远来看,执行的频率将正好是指定周期的倒数(假定Object.wait(long)
所依靠的系统时钟是准确的)。
固定速率执行适用于那些对绝对时间敏感的重复执行活动,如每小时准
点打钟报时,或者在每天的特定时间运行已安排的维护活动。它还适用于
那些完成固定次数执行的总计时间很重要的重复活动,如倒计时的计时
器,每秒钟滴答一次,共10秒钟。最后,固定速率执行适用于安排多个
重复执行的计时器任务,这些任务相互之间必须保持同步。
task -所要安排的任务。
delay -执行任务前的延迟时间,单位是毫秒。
period -执行各后续任务之间的时间间隔,单位是毫秒。
IllegalArgumentException -如果delay是负数, 或者delay +
System.currentTimeMillis() 是负数。
IllegalStateException -如果已经安排或取消了任务,已经取消了计时器,或者计时
器线程已终止。
scheduleAtFixedRate
public void scheduleAtFixedRate(TimerTasktask,
Date firstTime,
long period)
安排指定的任务在指定的时间开始进行重复的固定速率执行。以近似固定的时间间
隔(由指定的周期分隔)进行后续执行。
在固定速率执行中,相对于已安排的初始执行时间来安排每次执行。如果
由于任何原因(如垃圾回收或其他背景活动)而延迟了某次执行,则将快
速连续地出现两次或更多次执行,从而使后续执行能够赶上来。从长远来
看,执行的频率将正好是指定周期的倒数(假定Object.wait(long)所
依靠的系统时钟是准确的)。
固定速率执行适用于那些对绝对时间敏感的重复执行活动,如每小时准
点打钟报时,或者在每天的特定时间运行已安排的维护活动。它还适用于
那些完成固定次数执行的总计时间很重要的重复活动,如倒计时的计时
器,每秒钟滴答一次,共10秒钟。最后,固定速率执行适用于安排多次
重复执行的计时器任务,这些任务相互之间必须保持同步。
task -所要安排的任务。
firstTime -首次执行任务的时间。
period -执行各后续任务之间的时间间隔,单位是毫秒。
IllegalArgumentException -如果time.getTime()是负数。
IllegalStateException -如果已经安排或取消了任务,已经取消了计时器,或者计时
器线程已终止。
public void cancel()
终止此计时器,丢弃所有当前已安排的任务。这不会干扰当前正在执行的任务(如
果存在)。一旦终止了计时器,那么它的执行线程也会终止,并且无法根据它安排更
多的任务。
注意,在此计时器调用的计时器任务的run方法内调用此方法,就可以
绝对确保正在执行的任务是此计时器所执行的最后一个任务。
可以重复调用此方法;但是第二次和后续调用无效。
public int purge()
从此计时器的任务队列中移除所有已取消的任务。调用此方法对计时器的行为没有
影响,但是将无法引用队列中已取消的任务。如果没有对这些任务的外部引用,则
它们就成为垃圾回收的合格对象。
多数程序无需调用此方法。它设计用于一些罕见的应用程序,这些程序可
取消大量的任务。调用此方法要以时间来换取空间:此方法的运行时可能
与n + c log n呈正比,其中n是队列中的任务数,而c
是取消的任
注意,从此计时器上所安排的任务中调用此方法是允许的。
从队列中移除的任务数。
从以下版本开始:
下面是TimerTask类的介绍
类TimerTask
java.lang.Object
java.util.TimerTask
所有已实现的接口:
public abstract class TimerTask
extends Object
implements Runnable
由Timer安排为一次执行或重复执行的任务。
从以下版本开始:
另请参见:
构造方法摘要
protected TimerTask()
创建一个新的计时器任务。
boolean cancel()
取消此计时器任务。
abstract void run()
此计时器任务要执行的操作。
long scheduledExecutionTime()
返回此任务最近实际执行的安排执行时间。
从类java.lang.Object继承的方法
equals, finalize,getClass,hashCode,
notify,notifyAll,toString,
wait,wait,wait
构造方法详细信息
protected TimerTask()
创建一个新的计时器任务。
方法详细信息
public abstract void run()
此计时器任务要执行的操作。
接口Runnable中的run
另请参见:
Thread.run()
public boolean cancel()
取消此计时器任务。如果任务安排为一次执行且还未运行,或者尚未安排,则永远
不会运行。如果任务安排为重复执行,则永远不会再运行。(如果发生此调用时任务
正在运行,则任务将运行完,但永远不会再运行。)
注意,从重复的计时器任务的run方法中调用此方法绝对保证计时器任
务永远不会再运行。
此方法可以反复调用;第二次和以后的调用无效。
如果此任务安排为一次执行且尚未运行,或者此任务安排为重复执行,则返回true。
如果此任务安排为一次执行且已经运行,或者此任务尚未安排,或者此任务已经取
消,则返回false。(一般来说,如果此方法阻止发生一个或多个安排执行,则返回
scheduledExecutionTime
public long scheduledExecutionTime()
返回此任务最近实际执行的安排执行时间。(如果在任务执行过程中调用此方法,
则返回值为此任务执行的安排执行时间。)
通常从一个任务的run方法中调用此方法,以确定当前任务执行是否能
充分及时地保证完成安排活动:
public void run() {
if (System.currentTimeMillis() - scheduledExecutionTime()
MAX_TARDINESS)
// T skip this execution.
// Perform the task
通常,此方法不与固定延迟执行的重复任务一起使用,因为其安排执行时间允许
随时间浮动,所以毫无意义。
最近发生此任务执行安排的时间,采用Date.getTime()返回的格式。如果任务已开
始其首次执行,则返回值不确定。
另请参见:
Date.getTime()
本文已收录于以下专栏:
相关文章推荐
在android开发中,我们常常需要用到计时器,倒计时多少秒后再执行相应的功能,下面我就分别来讲讲这三种常用的计时的方法。
一、CountDownTimer
该类是个抽象类,如果要使用这个类中的方...
java提供了一个很方便的timer类,该类在javax.swing包中。当某些操作需要周期性执行 就可以使用计时器。
我们可以使用Timer类的构造方法Timer(int a,Object b)创建...
import javax.swing.JF
import javax.swing.T
import java.awt.event.ActionL
多种方式实现倒计时定时器
github链接:/Al-assad/Java-Gadgets/tree/master/count_down
1、简易方式实现
应用名称:Java计时器
用到的知识:Java GUI编程
开发环境:win8+eclipse+jdk1.8
import javax.swing.*;...
现在项目中用到需要定时去检查文件是否更新的功能。timer正好用于此处。
用法很简单,new一个timer,然后写一个timertask的子类即可。
代码如下:
package comz.aut...
公司做项目,其中涉及到一个游戏的业务逻辑,需要使用定时类去执行,在使用Timer的时候,发现总是出现奇怪的问题。
如下代码:
package com.yifeng.test22;
import ...
这两个类使用起来非常方便,可以完成我们对定时器的绝大多数需求
Timer类是用来执行任务的类,它接受一个TimerTask做参数
Timer有两种执行任务的模式,最常用的是schedu...
java定时器的使用(Timer)1、在应用开发中,经常需要一些周期性的操作,比如每5分钟执行某一操作等。对于这样的操作最方便、高效的实现方式就是使用java.util.Timer工具类。privat...
java中自带的定时器有Timer和TimerTask,但是运行起来要结束任务用cancel方法可以结束该任务,此时Timer线程还在运行,程序并没有退出,那么怎么结束Timer呢?
示例代码如下p...
您举报文章:
举报原因:
原文地址:
原因补充:
(最多只允许输入30个字)

我要回帖

更多关于 类java 的文章

 

随机推荐