前几看了一下《thinking in java》了解到java原生的Times类有两个问题:

(1)Timer是启动单个线程来处理所有的时间任务,如果一个任务耗时很久,那么如果在执行这个过程中,下个定时任务开始,就会对接下来的任务造成影响;

(2)Timer某一个定时程序在执行过程中抛出运行时异常,那么定时器就会以为终止定时器的运行;

首先了解一下Timer类的核心组成

Timer有两个核心的属性,一个是TaskQueue对象,用于存储任务队列,一个是TimerThread,用于执行队列中的任务。

Java代码  

  1. public class Timer {

  2. private TaskQueue queue = new TaskQueue();

  3. private TimerThread thread = new TimerThread(queue);

  4. 。。。。。。

写代码验证一下,看看是否是这样的。

Timer和TimeTask是java基础中实现定时程序的两个核心类。

A、TimerTask是一个抽象类,内部实现了Runnable接口,同时定义了任务的几个状态;

B、Timer类来负责调度继承TimerTask的具体任务,可是实现定时或者指定周期运行

通过打印可以发现,代码是顺序执行的,最简单的代码实现:

Java代码  

  1. public class TimerTest extends TimerTask{

  2. public static void main(String[] args) {

  3. Timer timer1 = new Timer("time1");

  4. timer1.schedule(new TimerTest(), 1);

  5. timer1.schedule(new TimerTest(), 2);

  6. timer1.schedule(new TimerTest(), 3);

  7. }

  8. @Override

  9. public void run() {

  10. try {

  11. System.out.println(Thread.currentThread().getName()+"-线程暂停5秒");

  12. TimeUnit.SECONDS.sleep(5);

  13. System.out.println(Thread.currentThread().getName()+"-线程执行结束");

  14. } catch (InterruptedException e) {

  15. e.printStackTrace();

  16. }

  17. }

  18. }

当代码运行中,第一个任务出现NullPointException时,第二个任务就不会再运行下去。

Java代码  

  1. public class TimerTest extends TimerTask{

  2. public static void main(String[] args) {

  3. Timer timer1 = new Timer("time1");

  4. timer1.schedule(new TimerTest(), 1);

  5. timer1.schedule(new TimerTest(), 2);

  6. }

  7. @Override

  8. public void run() {

  9. try {

  10. System.out.println(Thread.currentThread().getName()+"-线程暂停5秒");

  11. TimeUnit.SECONDS.sleep(5);

  12. System.out.println(Thread.currentThread().getName()+"-线程执行结束");

  13. String aaa = null;

  14. System.out.println(aaa.hashCode());

  15. } catch (InterruptedException e) {

  16. e.printStackTrace();

  17. }

  18. }

  19. }

Timer有三部分组成:

任务单元,任务队列,任务调度器,从下图能够非常直观的看出几个类直接的关系。

TaskQueue任务队列的管理:

一个具有优先级的时间任务队列,通过下次执行时间来进行排序。实际上就是一个数组,数据里面放着了TimeTask对象。

Java代码  

  1. void add(TimerTask task) {

  2. if (size + 1 == queue.length)

  3. queue = Arrays.copyOf(queue, 2*queue.length);

  4. <span style="white-space: pre;"> </span>    queue[++size] = task;

  5. fixUp(size);

  6. }

上面是TaskQueue中的add方法,添加新的TimeTask的时候,首先判断是否超出了数组长度,初始数组是128,如果超过,就将数组长度加倍。

最后执行fixUp,重新将数组进行排序,TimeTask中nextExecutionTime小的排在数组的最前面,即最近一个要执行的定时任务。

Timer中重载了这个方法schedule,用来实现不同场景的任务调度,每一个schedule方法中都是用了这个方法sched

Java代码  

  1. private void sched(TimerTask task, long time, long period) {

  2. if (time < 0)

  3. throw new IllegalArgumentException("Illegal execution time.");

  4. synchronized(queue) {

  5. if (!thread.newTasksMayBeScheduled)

  6. throw new IllegalStateException("Timer already cancelled.");

  7. synchronized(task.lock) {

  8. if (task.state != TimerTask.VIRGIN)

  9. throw new IllegalStateException(

  10. "Task already scheduled or cancelled");

  11. task.nextExecutionTime = time;

  12. task.period = period;

  13. task.state = TimerTask.SCHEDULED;

  14. }

  15. queue.add(task);

  16. if (queue.getMin() == task)

  17. queue.notify();

  18. }

  19. }

从上述代码中可以看出,首先锁住任务队列,如果task符合条件,则直接将任务加入到任务队列中,并设置好任务下次执行时间以及执行周期并设置状态为可以被调度。

鉴于 Timer 的上述缺陷,Java 5 推出了基于线程池设计的 ScheduledExecutor。其设计思想是,每一个被调度的任务都会由线程池中一个线程去执行,因此任务是并发执行的,相互之间不会受到干扰。

java.util.courrent包中有一个定时调度线程的类:ScheduledExecutorService

下面的这个例子,从执行结果来看,两个定时程序是并发执行的,并且一个定时程序执行如果抛出异常,对于下个任务的执行没有啥影响,照样执行下去。

Java代码  

public class CourrentTimer implements Runnable {

Java代码  

  1. public static void main(String[] args) {

  2. CourrentTimer timer = new CourrentTimer();

  3. ScheduledExecutorService executer = Executors.newScheduledThreadPool(2);

  4. //创建线程池,调用ThreadPoolExecutor来进行创建,通过BlockingQueue来管理任务队列

  5. executer.submit(timer);

  6. executer.submit(timer);

  7. //Timer类实现Runnable接口,从而进入任务队列

  8. }

  9. @Override

  10. public void run() {

  11. try {

  12. TimeUnit.SECONDS.sleep(5);

  13. System.out.println(Thread.currentThread().getName()+"-线程暂停5秒后运行-"+new Date());

  14. String aa = null;

  15. System.out.println(aa.hashCode());

  16. } catch (Exception e) {

  17. e.printStackTrace();

  18. }

  19. }

下面是执行返回的结果

pool-1-thread-1-线程暂停5秒后运行-Mon Feb 13 20:54:06 CST 2012
java.lang.NullPointerException
at CourrentTimer.run(CourrentTimer.java:24)
at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:441)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
at java.util.concurrent.FutureTask.run(FutureTask.java:138)
at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.access$301(ScheduledThreadPoolExecutor.java:98)
at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:206)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
pool-1-thread-2-线程暂停5秒后运行-Mon Feb 13 20:54:06 CST 2012
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:662)
java.lang.NullPointerException
at CourrentTimer.run(CourrentTimer.java:24)
at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:441)
at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:303)
at java.util.concurrent.FutureTask.run(FutureTask.java:138)
at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.access$301(ScheduledThreadPoolExecutor.java:98)
at java.util.concurrent.ScheduledThreadPoolExecutor$ScheduledFutureTask.run(ScheduledThreadPoolExecutor.java:206)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:662)

java的Timer和TimerTask两个类以及java.util.courrent中的ScheduledExecutorService可是实现定时以及周期的执行,但是复杂的定时任务(例如每周二晚上0点到4点执行)就暂时无法满足,需要结果时间工具类,Spring quartz将定时程序作了一个很好的封装,使开发者通过简单的配置就可以实现定时程序的调度。

下面介绍几个Quartz的核心概念

主要解决了一个问题:

定时任务调度器,调用什么任务,啥时候去调用。

1、Job
表示一个工作,要执行的具体内容。此接口中只有一个方法
void execute(JobExecutionContext context)

2、JobDetail
JobDetail表示一个具体的可执行的调度程序,Job是这个可执行程调度程序所要执行的内容,另外JobDetail还包含了这个任务调度的方案和策略。

3、Trigger代表一个调度参数的配置,什么时候去调。

4、Scheduler代表一个调度容器,一个调度容器中可以注册多个JobDetail和Trigger。当Trigger与JobDetail组合,就可以被Scheduler容器调度了。

Java代码  

  1. public interface Job {

  2. //在Trigger配置下通过scheduler来进行调度的具体任务

  3. public void execute(JobExecutionContext context)

  4. throws JobExecutionException;

  5. //任务的上下文对象,包含了任务执行细节中的各个情况

  6. }

Spring和Quartz的整合,可以通过简单扼配置,就能完成负责的时间调度任务,对于应用来说,只需要实现具体的任务即可,时间的调度等等细节交由Quartz来执行。

下面通过一个简单的例子来说明:

<!--
定义调用对象和调用对象的方法,这里定义了执行任务的bean以及bean中的方法
-->
<bean id="jobtask" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
<property name="targetBeanName">
<value>executerInitCache</value>
</property>
<property name="targetMethod">
<value>load</value>
</property>
</bean>

<!-- 定义触发时间,相当于抽象了任务执行的规则,这个Trigger类需要具体的job以及关联这个job的时间规则 -->
<bean id="doTime" class="org.springframework.scheduling.quartz.CronTriggerBean">
<property name="jobDetail">
<ref bean="jobtask" />
</property>
<property name="cronExpression">
<value>0 30 2 * * ? *</value>
</property>
</bean>

<!-- 总管理类 如果将lazy-init='false'那么容器启动就会执行调度程序 ,任务的调度器 -->
<bean id="startQuertz" lazy-init="false" autowire="no" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers">
<list>
<ref bean="doTime" />
</list>
</property>
</bean>

timeTask俗称TT,是淘宝内部封装的一个时间调度器,Spring Quartz可以快速的完成时间的调度,但是如果应用是在集群环境下,我们就希望有一台机器来完成定时的任务,如果每台机器跑的代码完成一样,那么这个定时器就会在每台机器上都跑一边,如果跑的业务流程是幂等的还好,如果不是,那就会很多问题,TT能够实现的一个简单功能是,定时程序可以通过后台页面点击触发,能够制定集群中的一台机器来运行,具体原理介绍一下。

综合上面的一些结论,受到的一些启发:

(1)抽象任务接口,通过实现任务接口,来满足各种各样任务的需要,从而满足业务场景
(2)任务队列,存放实现任务接口的业务任务
(3)任务调度单独抽象,用来调度任务的执行
(4)抽象规则,定时任务有个规则就是定时程序啥时候执行,将这个设置抽象成一个简单的设置,通过一些表达式,来完成,简单高效

文章来源:http://iamzhongyong.iteye.com/blog/1404149

【转载】Java定时器的学习的更多相关文章

  1. (转载) java:IO流学习小结

    今天刚刚看完Java的io流操作,把主要的脉络看了一遍,不能保证以后使用时都能得心应手,但是最起码用到时知道有这么一个功能可以实现,下面对学习进行一下简单的总结: IO流主要用于硬板.内存.键盘等处理 ...

  2. Java I/O学习

    转载: Java I/O学习 一.Java I/O类结构以及流的基本概念 在阅读Java I/O的实例之前我们必须清楚一些概念,我们先看看Java I/O的类结构图: Java I/O主要以流的形式进 ...

  3. [转载]Java学习这七年

    从2005那会做自动化测试开始接触Java开始,至今近7年.今天正好项目结束,趁机整理下思路,确定后续方向. 前三个年头基本上集中于Java基础的学习,包括设计模式,从完全不懂,到看的懂但似乎又不懂, ...

  4. 转载:一位资深程序员大牛给予Java初学者的学习路线建议

    一位资深程序员大牛给予Java初学者的学习路线建议   java学习这一部分其实也算是今天的重点,这一部分用来回答很多群里的朋友所问过的问题,那就是我你是如何学习Java的,能不能给点建议?今天我是打 ...

  5. (转载)一位资深程序员大牛给予Java初学者的学习建议

    这一部分其实也算是今天的重点,这一部分用来回答很多群里的朋友所问过的问题,那就是我你是如何学习Java的,能不能给点建议? 今天我是打算来点干货,因此咱们就不说一些学习方法和技巧了,直接来谈每个阶段要 ...

  6. java之jvm学习笔记二(类装载器的体系结构)

    java的class只在需要的时候才内转载入内存,并由java虚拟机的执行引擎来执行,而执行引擎从总的来说主要的执行方式分为四种, 第一种,一次性解释代码,也就是当字节码转载到内存后,每次需要都会重新 ...

  7. Java IO流学习总结三:缓冲流-BufferedInputStream、BufferedOutputStream

    Java IO流学习总结三:缓冲流-BufferedInputStream.BufferedOutputStream 转载请标明出处:http://blog.csdn.net/zhaoyanjun6/ ...

  8. Java IO流学习总结八:Commons IO 2.5-IOUtils

    Java IO流学习总结八:Commons IO 2.5-IOUtils 转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/550519 ...

  9. 聊聊阿里社招面试,谈谈“野生”Java程序员学习的道路

    引言 很尴尬的是,这个类型的文章其实之前笔者就写过,原文章里,笔者自称LZ(也就是楼主,有人说是老子的简写,笔者只想说,这位同学你站出来,保证不打死你,-_-),原文章名称叫做<回答阿里社招面试 ...

随机推荐

  1. 1通过URL对象的openStream()方法能够得到指定资源的输入流。

    通过URL读取网页内容     1通过URL对象的openStream()方法能够得到指定资源的输入流.     2通过输入流能够读取.訪问网络上的数据.     案例: import java.io ...

  2. Eclipse 使用 SVN 插件后改动用户方法汇总

    判定 SVN 插件是哪个 JavaH 的处理方法 SVNKit 的处理方法 工具自带改动功能 删除缓存的秘钥文件 其他发表地点 判定 SVN 插件是哪个 常见的 Eclipse SVN 插件我知道的一 ...

  3. 谈谈PHP网站的防SQL注入

    SQL(Structured Query Language)即结构化查询语言.SQL 注入,就是把 SQL 命令插入到 Web 表单的输入域或页面请求参数的查询字符串中,在 Web表单向 Web 服务 ...

  4. MySQL + Amoeba 负载均衡、主从备份方案

    1.  基本环境 4台内网虚拟机的操作系统都是ubuntu-14.04.4 64位 IP为:192.168.169.11.192.168.169.12.192.168.169.13.192.168.1 ...

  5. Excel2007制作直方图和正态分布曲线图

    对同一维度的数据分析数据分布范围及分布趋势,要通过制作直方图和正态分布曲线图体现. 例如:已知所有员工的日收入,分析员工收入分布情况(51.7,50.6,57.9,56.9,56.7,56.7,55. ...

  6. window.name实现跨域数据传输

    偶然间碰到个问题,通过JS给window.name赋值数组情况下,在firefox与chrome下会转换为字符串类型,在IE11下则显示正常.不说了,上图(firefox下): 代码: <scr ...

  7. Linux的文件传输命令总结

    由于工作原因,须要常常在不同的server见进行文件传输,特别是大文件的传输,因此对linux下不同server间传输数据命令和工具进行了研究和总结.主要是rcp,scp,rsync,ftp,sftp ...

  8. 为php添加pcntl扩展,多线程

    前言: pcntl 介绍 pcntl扩展可以支持 PHP 的多线程操作.(非Unix类系统不支持此模块) phpize 介绍 phpize 可以用来给 PHP 动态的添加扩展.比如编译 PHP 时忘记 ...

  9. Junit 内部解密之一: Test + TestCase + TestSuite

    转自:http://blog.sina.com.cn/s/blog_6cf812be0100wbhq.html nterface: Test 整个测试的的基础接口 Method 1: abstract ...

  10. MySQL_使用时遇到的问题汇总

    一.data too long for column 'name' at row 1 1.现象:把数据库的字符集编码设置为utf-8,通过DOS界面向表的某一列插入汉字时会遇到类似 data too ...