zoukankan      html  css  js  c++  java
  • Java的定时器总结(包含普通写法与Spring家族框架配置写法)

        工作中会时长用到定时器,无论是任务的固定时间执行还是任务执行频率设置,都是需要定时器的,在此将java中的定时器进行一次总结整理

      非maven项目的定时器(三种写法)

    (1):创建thread,通过重写线程run方法来实现定时执行任务

    /**
     * 普通thread
     * 这是最常见的,创建一个thread,然后让它在while循环里一直运行着,
     * 通过sleep方法来达到定时任务的效果。这样可以快速简单的实现,代码如下:
     * @author GT
     *
     */
    public class Task1 {
        public static void main(String[] args) {
            // run in a second
            final long timeInterval = 1000;
            Runnable runnable = new Runnable() {
                public void run() {
                    while (true) {
                        // ------- code for task to run
                        System.out.println("Hello !!");
                        // ------- ends here
                        try {
                            Thread.sleep(timeInterval);
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    }
                }
            };
            Thread thread = new Thread(runnable);
            thread.start();
        }
    }

    (2):Timer类可以调度任务,TimerTask则是通过在run()方法里实现具体任务。 Timer实例可以调度多任务,它是线程安全的。

    import java.util.Timer;
    import java.util.TimerTask;
     
    /**
     * 
     * 于第一种方式相比,优势 1>当启动和去取消任务时可以控制 2>第一次执行任务时可以指定你想要的delay时间
     * 
     * 在实现时,Timer类可以调度任务,TimerTask则是通过在run()方法里实现具体任务。 Timer实例可以调度多任务,它是线程安全的。
     * 当Timer的构造器被调用时,它创建了一个线程,这个线程可以用来调度任务。 下面是代码:
    */
    public class Task2 {
        public static void main(String[] args) {
            TimerTask task = new TimerTask() {
                @Override
                public void run() {
                    // task to run goes here
                    System.out.println("Hello !!!");
                }
            };
            Timer timer = new Timer();
            long delay = 0;
            long intevalPeriod = 1 * 1000;
            // schedules the task to be run in an interval
            timer.scheduleAtFixedRate(task, delay, intevalPeriod);
        } // end of main
    }

    (3):

    import java.util.concurrent.Executors;
    import java.util.concurrent.ScheduledExecutorService;
    import java.util.concurrent.TimeUnit;
     
    /**
     * 
     * 
     * ScheduledExecutorService是从Java SE5的java.util.concurrent里,做为并发工具类被引进的,这是最理想的定时任务实现方式。 
     * 相比于上两个方法,它有以下好处:
     * 1>相比于Timer的单线程,它是通过线程池的方式来执行任务的 
     * 2>可以很灵活的去设定第一次执行任务delay时间
     * 3>提供了良好的约定,以便设定执行的时间间隔
     * 
     */
    public class Task3 {
        public static void main(String[] args) {
            Runnable runnable = new Runnable() {
                public void run() {
                    // task to run goes here
                    System.out.println("Hello !!");
                }
            };
            ScheduledExecutorService service = Executors
                    .newSingleThreadScheduledExecutor();
            // 第二个参数为首次执行的延时时间,第三个参数为定时执行的间隔时间
            service.scheduleAtFixedRate(runnable, 10, 1, TimeUnit.SECONDS);
        }
    }

    Maven框架中Spring家族的定时器

    1. spring学习系列 -- 定时器一TimerTask

    spring定时器一般有两种:

    TimerTask、Quartz。本节只讲TimerTask

    需要的包:

    aopalliance-1.0.jar

    commons-logging-1.1.1.jar

    spring-aop-3.0.6.RELEASE.jar

    spring-asm-3.0.6.RELEASE.jar

    spring-beans-3.0.6.RELEASE.jar

    spring-context-3.0.6.RELEASE.jar

    spring-core-3.0.6.RELEASE.jar

    spring-expression-3.0.6.RELEASE.jar

    TimerTask实例:同时启动2个定时器,执行任务

    定时执行任务的类继承TimerTask:

    Java代码

    public class EmailReportTask extends TimerTask{     
        @Override     
        public void run() {     
            System.out.println(" EmailReportTask Run... ");  
        }       
    }   
      
    public class PageReportTask extends TimerTask{     
        @Override     
        public void run() {     
            System.out.println("PageReportTask Run...");  
        }       
    }     

    spring的配置文件:

    Xml代码

     <!--  Bean  -->   
    <bean id="emailReportTask" class="com.hry.spring.timertask.EmailReportTask" />  
    <bean id="pageReportTask" class="com.hry.spring.timertask.PageReportTask" />  
      
    <!-- ScheduledTimerTask设置定时器属性 : period=定时器周期;delay=延迟多久启动   
         86400000代表24个小时;timerTask=执行定时任务的类对象  -->   
    <bean id="emailReportScheduleReportTask"   
        class="org.springframework.scheduling.timer.ScheduledTimerTask">     
        <property name="timerTask" ref="emailReportTask" />     
        <property name="period" value="2000" />     
        <property name="delay" value="1000" />     
    </bean>   
    <bean id="pageReportScheduleReportTask"   
        class="org.springframework.scheduling.timer.ScheduledTimerTask">     
        <property name="timerTask" ref="pageReportTask" />     
        <property name="period" value="2000" />      
    </bean>   
      
        <!-- Spring的TimerFactoryBean负责启动定时任务;   
            scheduledTimerTasks = 需要启动的定时器任务的列表-->  
        <bean class="org.springframework.scheduling.timer.TimerFactoryBean">     
        <property name="scheduledTimerTasks">     
            <list>  
                <ref bean="emailReportScheduleReportTask"/>  
                <ref bean="pageReportScheduleReportTask"/>  
            </list>    
        </property>     
    </bean>   

    
    

    /*测试代码:

    Java代码*/

    public class TestBaseService {  
        protected ApplicationContext ctx = new ClassPathXmlApplicationContext(  
                new String[]{  
                        "classpath:resource/spring.xml"  
                }  
        );  
      
        @Test  
        public void timer(){  
            try{  
                // 这个是主线程,如果结束了,则定时器也会结束,所有设置时间要长  
                Thread.sleep(36 * 1000);  
            }catch(Exception e){  
                e.printStackTrace();  
            }  
       
        }  
    }  

    2.Spring的第二种定时器quartz
    在上一节的基础上,讨论Spring的第二种定时器quartz

      重点包: 

    quartz-1.8.6.jar

    org.springframework.context.support-3.1.1.RELEASE.jar

      包说明:

            quartz包请使用1.8.6或以下的版本,因为quartz2.0版本和spring3.1.1存在冲突,会抛出

    目的:

       通过quartz定时循环执行一个任务

    任务类:

        该类还包含一个属性sTest及其set/get方法

    Java代码

    public class MyJob extends QuartzJobBean {  
        private String sTest;  
      
        @Override  
        protected void executeInternal(JobExecutionContext job)  
                throws JobExecutionException {  
            // TODO Auto-generated method stub  
            System.out.println("sTest = " + sTest);  
            System.out.println("MyJob Run...");  
        }  
      
        public String getsTest() {  
            return sTest;  
        }  
      
        public void setsTest(String sTest) {  
            this.sTest = sTest;  
        }  
          
    }  

    spring的配置文件

        JobDetailBean:设置要执行任务的类,在这里通过jobDataAsMap属性还可以设置任务类的属性。

      Trigger用于设置人物类启动的时间,循环的间隔等工作信息。spring包含2种Trigger :

    simpleReportTrigger功能类似上一篇文章的TimerTask,设置任务延迟多久启动,循环间隔等信息

    cronReportTrigger可以设置任务精确工作的时间

      SchedulerFactoryBean:仅仅有以上2个类设置是不够的,如果要启动定时器,还需要通过此类设置要启动的trigger。

    Xml代码

    <!--   
        JobDetailBean是Quartz的org.quartz.JobDetail的子类,它要求通过jobClass属性来设置一个Job对象。  
        jobClass = 要执行定时任务的类  
        jobDataAsMap = 用于向任务类对象中注入信息,即可以注入值,也可以引用另一个变量  
         -->  
        <bean id="reportJob" class="org.springframework.scheduling.quartz.JobDetailBean">     
        <property name="jobClass" >     
            <value>com.hry.spring.timertask.MyJob</value>     
        </property>     
        <property name="jobDataAsMap">     
            <map>     
                <entry key="sTest">    
                    <value>10</value>   
                </entry>     
                <!--    
                <entry key="courseService">     
                    <ref bean="courseService"/>     
                </entry>   
                -->  
            </map>     
        </property>     
    </bean>  
        <!--   
            org.quartz.Trigger用于设置任务类如何工作  
            SimpleTriggerBean 类似 ScheduledTimerTask,配置定时器的执行频率和延迟多久执行  
            jobDetail = 实际工作类  
            startDelay = 延迟实际  
            repeatInterval = 重复频率  
         -->  
        <bean id="simpleReportTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">     
        <property name="jobDetail" ref="reportJob" />     
        <property name="startDelay" value="1000" />     
        <property name="repeatInterval" value="2000" />     
        </bean>  
          
        <!--   
            CronTriggerBean 指定某个时间允许任务  
            cronExpression = 通过表达式设置特定的时间点执行  
        -->  
    <bean id="cronReportTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">     
        <property name="jobDetail" ref="reportJob" />     
        <property name="cronExpression" value = "40 * * * * ?" />     
    </bean>    
          
          
        <!--   
            SchedulerFactoryBean:启动定时器  
            triggers = 启动哪些定时任务,可以多个  
         -->  
        <bean class="org.springframework.scheduling.quartz.SchedulerFactoryBean">     
        <property name="triggers">     
                <list>  
                    <ref bean="simpleReportTrigger"/>  
                <!--  
                    <ref bean="cronReportTrigger" /> 
                    -->  
                </list>     
        </property>     
    </bean>     

    通过上篇文章的测试代码执行以上程序,则有如下输入,说明我们程序成功了。

    Html代码

     sTest = 10  

    MyJob Run...  
    sTest = 10  
    MyJob Run...  
    sTest = 10  
    MyJob Run...  
    sTest = 10  
    MyJob Run...  
    sTest = 10  
    MyJob Run...  

    CronTriggerBean的精华是cron表达式,以下是关于cron表达式(来自网络):

    Cron 表达式依照顺序有7 个字段:

    小时

    月内日期

    周内日期

    年(可选字段)

    特殊字符

    Cron 触发器利用一系列特殊字符,如下所示:

    反斜线(/)字符表示增量值。例如,在秒字段中“5/15”代表从第 5 秒开始,每 15 秒一次。

    问号(?)字符和字母 L 字符只有在月内日期和周内日期字段中可用。问号表示这个字段不包含具体值。所以,如果指定月内日期,可以在周内日期字段中插入“?”,表示周内日期值无关紧要。字母 L 字符是 last 的缩写。放在月内日期字段中,表示安排在当月最后一天执行。在周内日期字段中,如果“L”单独存在,就等于“7”,否则代表当月内周内日期的最后一个实例。所以“0L”表示安排在当月的最后一个星期日执行。 

    在月内日期字段中的字母(W)字符把执行安排在最靠近指定值的工作日。把“1W”放在月内日期字段中,表示把执行安排在当月的第一个工作日内。

    井号(#)字符为给定月份指定具体的工作日实例。把“MON#2”放在周内日期字段中,表示把任务安排在当月的第二个星期一。

    星号(*)字符是通配字符,表示该字段可以接受任何可能的值。

    字段 允许值 允许的特殊字符 

    秒 0-59 , - * / 

    分 0-59 , - * / 

    小时 0-23 , - * / 

    日期 1-31 , - * ? / L W C 

    月份 1-12 或者 JAN-DEC , - * / 

    星期 1-7 或者 SUN-SAT , - * ? / L C # 

    年(可选) 留空, 1970-2099 , - * /

    表达式意义 

    "0 0 12 * * ?" 每天中午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期间的每1分钟触发 

    "0 0/5 14 * * ?" 在每天下午2点到下午2:55期间的每5分钟触发 

    "0 0/5 14,18 * * ?" 在每天下午2点到2:55期间和下午6点到6:55期间的每5分钟触发 

    "0 0-5 14 * * ?" 在每天下午2点到下午2:05期间的每1分钟触发 

    "0 10,44 14 ? 3 WED" 每年三月的星期三的下午2:10和2:44触发 

    "0 15 10 ? * MON-FRI" 周一至周五的上午10:15触发 

    "0 15 10 15 * ?" 每月15日上午10:15触发 

    "0 15 10 L * ?" 每月最后一日的上午10:15触发 

    "0 15 10 ? * 6L" 每月的最后一个星期五上午10:15触发 

    "0 15 10 ? * 6L 2002-2005" 2002年至2005年的每月的最后一个星期五上午10:15触发 

    "0 15 10 ? * 6#3" 每月的第三个星期五上午10:15触发 

    每天早上6点

    0 6 * * *

    每两个小时

    0 */2 * * * 

    晚上11点到早上8点之间每两个小时,早上八点

    0 23-7/2,8 * * *

    每个月的4号和每个礼拜的礼拜一到礼拜三的早上11点

    0 11 4 * 1-3 

    1月1日早上4点

    0 4 1 1 *

    3.Spring的两种定时器的区别
    前2节绍了spring的两种定时器:TimerTask 和 Quartz,本节要在这两节的基础上,讲讲两者的区别。

    精确度和功能

       Quartz可以通过cron表达式精确到特定时间执行,而TimerTask不能。Quartz拥有TimerTask所有的功能,而TimerTask则没有。

    任务类的数量

        TimerTask和Quartz每次执行任务时,每次调用的是不是都是同一个任务类对象,还是每次都不一样?现在做如下实验,每次执行任务时,将任务类对象本身打印出来。

    Quartz任务类

    Java代码

    public class MyJob extends QuartzJobBean {  
        private String sTest;  
      
        @Override  
        protected void executeInternal(JobExecutionContext job)  
                throws JobExecutionException {  
            // TODO Auto-generated method stub  
            System.out.println("sTest = " + sTest);  
            System.out.println("MyJob Run..." + this);  
        }  
      
        // set/get 略  
          
    }  

    Quartz输出结果

    Java代码

    sTest = 10  
    MyJob Run...com.hry.spring.timertask.MyJob@1060478  
    sTest = 10  
    MyJob Run...com.hry.spring.timertask.MyJob@db4fa2  
    sTest = 10  
    MyJob Run...com.hry.spring.timertask.MyJob@491c4c  

    从输出结果可以看出,Quartz每次执行都创建一个新的任务类对象。

    TimerTask任务类

    public class EmailReportTask extends TimerTask{     
        // 每次执行过程中num的值都会发生变化,说明此事使用的是同一个类对象  
        private int num = 0;  
        @Override     
        public void run() {     
            System.out.println("num = " + num++);  
            System.out.println(this);  
        }       
    }    

    从输出结果可以看出,TimerTask每次执行时,都是使用同一个对象

     从以上的分析,可以得出结论:Quartz每次执行任务都创建一个新的任务类对象,而TimerTask则每次使用同一个任务类对象。

    对异常的处理

    一个循环执行的任务,如果某一次执行任务时,因为某些原因抛出异常,则定时器是否还会在下一个执行任务的时间点执行任务吗?下面通过模拟在任务类中抛出异常,来模拟这种情况,并测试两种定时器如何处理这种情况。

    Quartz任务类

    public class MyJob extends QuartzJobBean {  
        private String sTest;  
      
        @Override  
        protected void executeInternal(JobExecutionContext job)  
                throws JobExecutionException {  
            // TODO Auto-generated method stub  
            System.out.println("sTest = " + sTest);  
            System.out.println("MyJob Run..." + this);  
            throw new RuntimeException("Test");  
        }  
      
        // set/get 方法略  
          
    }  

    Quartz输出结果

    Html代码

    sTest = 10  
    MyJob Run...com.hry.spring.timertask.MyJob@16f25a7  
    2013-01-05 19:58:37,381 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-3] ERROR [org.quartz.core.JobRunShell] - Job DEFAULT.reportJob threw an unhandled Exception:   
      java.lang.RuntimeException: Test  
        at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)  
        at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)  
        at org.quartz.core.JobRunShell.run(JobRunShell.java:223)  
        at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)  
    2013-01-05 19:58:37,396 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-3] ERROR [org.quartz.core.ErrorLogger] - Job (DEFAULT.reportJob threw an exception.  
      org.quartz.SchedulerException: Job threw an unhandled exception. [See nested exception: java.lang.RuntimeException: Test]  
        at org.quartz.core.JobRunShell.run(JobRunShell.java:234)  
        at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)  
    Caused by: java.lang.RuntimeException: Test  
        at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)  
        at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)  
        at org.quartz.core.JobRunShell.run(JobRunShell.java:223)  
        ... 1 more  
    sTest = 10  
    MyJob Run...com.hry.spring.timertask.MyJob@110c31  
    2013-01-05 19:58:39,381 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-4] ERROR [org.quartz.core.JobRunShell] - Job DEFAULT.reportJob threw an unhandled Exception:   
      java.lang.RuntimeException: Test  
        at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)  
        at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)  
        at org.quartz.core.JobRunShell.run(JobRunShell.java:223)  
        at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)  
    2013-01-05 19:58:39,396 [org.springframework.scheduling.quartz.SchedulerFactoryBean#0_Worker-4] ERROR [org.quartz.core.ErrorLogger] - Job (DEFAULT.reportJob threw an exception.  
      org.quartz.SchedulerException: Job threw an unhandled exception. [See nested exception: java.lang.RuntimeException: Test]  
        at org.quartz.core.JobRunShell.run(JobRunShell.java:234)  
        at org.quartz.simpl.SimpleThreadPool$WorkerThread.run(SimpleThreadPool.java:549)  
    Caused by: java.lang.RuntimeException: Test  
        at com.hry.spring.timertask.MyJob.executeInternal(MyJob.java:24)  
        at org.springframework.scheduling.quartz.QuartzJobBean.execute(QuartzJobBean.java:113)  
        at org.quartz.core.JobRunShell.run(JobRunShell.java:223)  
        ... 1 more  

    以上结果中多次出现

    Java代码

    sTest = 10    
    MyJob Run...com.hry.spring.timertask.MyJob@110c31   
    可以看出,尽管每次执行任务时,任务类都会抛出异常,但是Quartz定时器,依然在下一个任务执行时间点执行任务,并没有因为异常,而导致定时器关闭,不再执行循环任务。

    TimerTask任务类

    Java代码

    num = 0  
    Exception in thread "org.springframework.scheduling.timer.TimerFactoryBean#0" java.lang.RuntimeException: test  
        at com.hry.spring.timertask.EmailReportTask.run(EmailReportTask.java:11)  
        at java.util.TimerThread.mainLoop(Unknown Source)  
        at java.util.TimerThread.run(Unknown Source)  

     TimerTask抛出异常后,后续再也没有执行此任务了,并且定时器所在的线程也自动结束。

    通过以上的分析,可以知道Quartz的某次执行任务过程中抛出异常,不影响下一次任务的执行,当下一次执行时间到来时,定时器会再次执行任务;而TimerTask则不同,一旦某个任务在执行过程中抛出异常,则整个定时器生命周期就结束,以后永远不会再执行定时器任务。

  • 相关阅读:
    python 读写json文件(dump, load),以及对json格式的数据处理(dumps, loads)
    Postman + newman + jenkins 的API自动化测试应用
    Selenium + WebDriver 各浏览器驱动下载地址
    pythonon ddt数据驱动二(json, yaml 驱动)
    python ddt 实现数据驱动一
    Python + logging 输出到屏幕,将log日志写入文件
    python 多线程小练习
    pycharm + git 的集成使用
    Map的遍历方式
    69道Spring面试题和答案
  • 原文地址:https://www.cnblogs.com/minixiong/p/11236943.html
Copyright © 2011-2022 走看看