java中Timer定時器的使用和啟動方式
Timer定時器的使用和啟動
1.概述
定時計劃任務功能在Java中主要使用的就是Timer對象,它在內部使用多線程的方式進行處理,所以它和多線程技術還是有非常大的關聯的。在JDK中Timer類主要負責計劃任務的功能,也就是在指定的時間開始執行某一個任務,但封裝任務的類卻是TimerTask類。
2.應用場景
我們使用timer的時候,一般有4種情況:
- 指定時間執行
- 指定時間執行後間隔指定時間重復執行
- 啟動任務之後多久執行
- 啟動任務後多久執行,執行之後指定間隔多久重復執行
3.使用方法
首先要通過繼承 TimerTask 類 並實現 run() 方法來自定義要執行的任務(當然也可以寫成匿名內部類),
需要創建一個定時器(Timer類對象),並通過Timer.schedule(TimerTask task,Date time) 方法執行時間運行任務
具體代碼如下:
package timerdemo; import java.util.Timer; import java.util.TimerTask; public class TimerDemo { public static void main(String[] args) { timerTest(); } public static void timerTest(){ //創建一個定時器 Timer timer = new Timer(); //schedule方法是執行時間定時任務的方法 timer.schedule(new TimerTask() { //run方法就是具體需要定時執行的任務 @Override public void run() { System.out.println("timer測試!!!"); } }, 1000, 10000); } }
這裡的 schedule方法有4個,分別對應上面說的四種情況:
4.啟動方法
1.在jar工程下啟動
把jar工程打成jar包,通過java -jar timer.jar 運行
2.這web工程下啟動
spring中我們可以通過實現接口ApplicationListener,並重寫public void onApplicationEvent(ApplicationEvent event) {}可以在容器初始話的時候執行這個方法
下面展示下web工程下每天00:00執行任務的代碼:
@Component public class SystemInitListener implements ApplicationListener<ContextRefreshedEvent> { @Override public void onApplicationEvent(ContextRefreshedEvent event) { //創建定時器 Timer timer = new Timer(); Calendar calendar = Calendar.getInstance(); calendar.add(Calendar.DATE,1); calendar.set(calendar.get(Calendar.YEAR),calendar.get(Calendar.MONTH),calendar.get(Calendar.DATE),0,0,0); long timeInterval = 24 * 60 * 60 * 1000; timer.schedule(new TimerTask() { @Override public void run() { // 每天00:00需要做的事情 } }, calendar.getTime(), timeInterval);
java的幾種定時器小結
總結一下我使用過的4種類型的定時器:@Scheduled註解、quartz、new Timer().schedule、使用線程控制。
1.@Scheduled註解
@Scheduled註解是最簡單的方式,隻需要啟用定時器,在方法上添加註解即可。
在spring配置中加入:
<!-- 啟用註解定時器 --> <task:annotation-driven />
在要具體的方法上加入註解@Scheduled
@Scheduled(cron = "0 0 * * * ? ") public void myTask(){ //定時任務...... }
2.quartz
quartz使用的是可配置的方式,將所有的定時器都配置再一個xml文件裡面。
步驟如下:
1.創建一個spring的配置文件:spring-quartz.xml
2.定義工作任務的job
3.定義觸發器Trigger並與job綁定
4.定義調度器,並將Trigger註冊到scheduler
<bean id="myTask" class="cn.coolwind.MyTask"/> <!-- 1.定義工作任務job --> <bean id="testJob" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean"> <!-- 定時器的類 --> <property name="targetObject" ref="myTask"></property> <!-- 需要定時執行的方法 --> <property name="targetMethod" value="test"></property> <property name="concurrent" value="false"></property> </bean> <!-- 2.定義觸發器Trigger並與Job綁定 --> <bean id="testJobTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean"> <property name="jobDetail" ref="testJob"/> <!-- 根據需要設置定時執行的時間 --> <property name="cronExpression" value="0 0/5 * * * ?" /> </bean> <!-- 3.定義調度器,並將trigger註冊進去 --> <bean name="quartzScheduler" class="org.springframework.scheduling.quartz.SchedulerFactoryBean"> <property name="triggers"> <list> <ref local="testJobTrigger" /> </list> </property> </bean>
最後記得將xml寫入web.xml裡!
<init-param> <param-name>contextConfigLocation</param-name> <param-value> classpath:applicationContext.xml, classpath:log4j.xml, classpath:spring-quartz.xml </param-value> </init-param>
3.使用Timer
使用Timer的schedule,schedule有3個參數:
schedule(TimerTask task, long delay, long period)
第一個為定時任務,根據業務需要重寫TimerTask的run方法即可;
第二個為延時啟動,單位毫秒;
第三個位多久運行一次,單位毫秒;
new Timer().schedule(new TimerTask() { @Override public void run() { try { //do Something } catch (Exception e) { e.printStackTrace(); } } },0,5L * 60 * 1000);
4.使用線程控制
使用線程來控制就更靈活一些,可以根據自己的需要判斷什麼時候運行,什麼時候停止,這需要對java的線程有一定的瞭解。
public class TaskTest { private static final ExecutorService pool = Executors.newFixedThreadPool(5);// 線程池 public static final TaskTest me = new TaskTest(); public final int[] arr = new int[]{1, 2, 3, 4, 5, 6, 7, 8, 9}; public static void main(String[] args) { me.start(); } private void start() { pool.execute(new Runnable() { @Override public void run() { while (true) { try { for (int i = 0; i < arr.length; i++) { if (1 == arr[i]) { System.out.println("start!"); Thread.sleep(1*1000L); } if (6 == arr[i]) { System.out.println("stop!"); Thread.sleep(5*1000L); } System.out.println(arr[i]); if (9 == arr[i]) { System.out.println("end!"); Thread.sleep(5*1000L); } } } catch (InterruptedException e) { e.printStackTrace(); } } } }); } }
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- java的幾種定時器的具體使用(4種)
- Java中定時任務的6種實現方式
- Java任務定時執行器案例的實現
- 淺析java中常用的定時任務框架-單體
- Java 實現訂單未支付超時自動取消功能(京東商城為例)