Android中使用Timer和TimerTask

前言

近期有个需求,要每隔一段时间,应用向后台发送一些数据,用作统计,这时可以使用Java提供的计时器的工具类,即Timer和TimerTask来实现这一功能。

简介

Timer是一个普通的类,其中有几个重要的方法;而TimerTask则是一个抽象类,其中有一个抽象方法run(),类似线程中的run()方法,我们使用Timer创建一个他的对象,然后调用schedule方法来完成这种间隔的操作。
这里用到的这个schedule方法有三个参数:
第一个参数就是TimerTask对象,我们实现TimerTask的run()方法就是要周期执行的一个任务;
第二个参数是延迟执行的时间,有两种类型,第一种是long类型,表示多长时间后开始执行,也就是说用户调用 schedule() 方法后,要等待这么长的时间才可以第一次执行run() 方法。另一种是Date类型,表示从那个时间后开始执行;
第三个参数就是执行的周期,为long类型,也就是说第一次调用之后,从第二次开始每隔多长的时间调用一次 run() 方法。

schedule方法还有一种两个参数的执行重载,第一个参数仍然是TimerTask,第二个表示为long的形式表示多长时间后执行一次,为Date就表示某个时间后执行一次。

Java API:

//Schedules thespecified task for execution after the specifieddelay
schedule(TimerTask task, long delay)//在延时delay毫秒后执行task。并没有提到重复执行

//Schedulesthe specified task for repeated fixed-delay execution, beginningafter the specified delay
schedule(TimerTask task, long delay, long period)//在延时delay毫秒后重复的执行task,周期是period毫秒。

Timer就是一个线程,使用schedule方法完成对TimerTask的调度,多个TimerTask可以共用一个Timer,也就是说Timer对象调用一次schedule方法就是创建了一个线程,并且调用一次schedule后TimerTask是无限制的循环下去的,使用Timer的cancel()停止操作。当然同一个Timer执行一次cancel()方法后,所有Timer线程都被终止。

另外Timer类还有一个调度方法scheduleAtFixedRate(),这个方法的参数与**schedule()**相同,具体的区别在下边再说!

使用

//true 说明这个timer以daemon方式运行(优先级低,程序结束timer也自动结束)   
Timer timer = new Timer(true);  
  
TimerTask task = new TimerTask() {  
   public void run() {  
   //每次需要执行的代码放到这里面。     
   }     
};  
  
//以下是几种调度task的方法:  
  
//time为Date类型:在指定时间执行一次。  
timer.schedule(task, time);  
  
//firstTime为Date类型,period为long,表示从firstTime时刻开始,每隔period毫秒执行一次。  
timer.schedule(task, firstTime, period);     
  
//delay 为long类型:从现在起过delay毫秒执行一次。  
timer.schedule(task, delay);  
  
//delay为long,period为long:从现在起过delay毫秒以后,每隔period毫秒执行一次。  
timer.schedule(task, delay, period);

两种调度方法的区别

①启动时间在指定时间之后(调用方法在设置的开始执行时间之后):
schedule()不会将过去的时间计算在内

SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
Date d1 = fTime.parse("2016/11/08 17:10:00");
timer.schedule(new TimerTask(){
	public void run(){
		System.out.println("this is task running");
	}
},d1,3*60*1000);

如果是在2016/11/08 17:20执行程序,则会在2016/11/08 17:23输出第一条:this is task running
scheduleAtFixedRate()会将过去的时间计算在内

SimpleDateFormat fTime = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss");
Date d1 = fTime.parse("2016/11/08 17:10:00");
timer.scheduleAtFixedRate(new TimerTask(){
	public void run(){
		System.out.println("this is task running");
	}
},d1,3*60*1000);

如果是在2016/11/08 17:20执行程序,则会快速输出三条:this is task running,第四条将会在2016/11/08 17:22输出,而不是 17:23!就是说是从指定的开始时间开始计时,而不是从执行时间开始计时
②任务延迟时的处理不同
schedule如果由于任何原因(如垃圾回收或其他后台活动)而延迟了某次执行,则后续执行也将被延迟。
schedule适用于那些需要“平稳”运行的重复活动。如以固定时间间隔闪烁的光标。
scheduleAtFixedRate如果由于任何原因(如垃圾回收或其他后台活动)而延迟了某次执行,则将快速连续地出现两次或更多的执行,从而使后续执行能够“追赶上来”。
scheduleAtFixedRate适用于那些对绝对时间敏感的重复执行活动,如每小时准点打钟报时,或者在每天的特定时间运行已安排的维护活动。

题外话
JDK 5.0以后推荐使用ScheduledThreadPoolExecutor,该类属于Executor Framework,它除了能处理异常外,还可以创建多个线程解决上面的问题。

参考文章:
Java中的Timer和TimerTask在Android中的用法
Timer 的 schedule()方法
Timer类的schedule和scheduleAtFixedRate 简单应用

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值