java中Timer定時(shí)器的使用和啟動(dòng)方式
Timer定時(shí)器的使用和啟動(dòng)
1.概述
定時(shí)計(jì)劃任務(wù)功能在Java中主要使用的就是Timer對(duì)象,它在內(nèi)部使用多線程的方式進(jìn)行處理,所以它和多線程技術(shù)還是有非常大的關(guān)聯(lián)的。在JDK中Timer類主要負(fù)責(zé)計(jì)劃任務(wù)的功能,也就是在指定的時(shí)間開(kāi)始執(zhí)行某一個(gè)任務(wù),但封裝任務(wù)的類卻是TimerTask類。
2.應(yīng)用場(chǎng)景
我們使用timer的時(shí)候,一般有4種情況:
- 指定時(shí)間執(zhí)行
- 指定時(shí)間執(zhí)行后間隔指定時(shí)間重復(fù)執(zhí)行
- 啟動(dòng)任務(wù)之后多久執(zhí)行
- 啟動(dòng)任務(wù)后多久執(zhí)行,執(zhí)行之后指定間隔多久重復(fù)執(zhí)行
3.使用方法
首先要通過(guò)繼承 TimerTask 類 并實(shí)現(xiàn) run() 方法來(lái)自定義要執(zhí)行的任務(wù)(當(dāng)然也可以寫成匿名內(nèi)部類),
需要?jiǎng)?chuàng)建一個(gè)定時(shí)器(Timer類對(duì)象),并通過(guò)Timer.schedule(TimerTask task,Date time) 方法執(zhí)行時(shí)間運(yùn)行任務(wù)
具體代碼如下:
package timerdemo; import java.util.Timer; import java.util.TimerTask; public class TimerDemo { public static void main(String[] args) { timerTest(); } public static void timerTest(){ //創(chuàng)建一個(gè)定時(shí)器 Timer timer = new Timer(); //schedule方法是執(zhí)行時(shí)間定時(shí)任務(wù)的方法 timer.schedule(new TimerTask() { //run方法就是具體需要定時(shí)執(zhí)行的任務(wù) @Override public void run() { System.out.println("timer測(cè)試!!!"); } }, 1000, 10000); } }
這里的 schedule方法有4個(gè),分別對(duì)應(yīng)上面說(shuō)的四種情況:
4.啟動(dòng)方法
1.在jar工程下啟動(dòng)
把jar工程打成jar包,通過(guò)java -jar timer.jar 運(yùn)行
2.這web工程下啟動(dòng)
spring中我們可以通過(guò)實(shí)現(xiàn)接口ApplicationListener,并重寫public void onApplicationEvent(ApplicationEvent event) {}可以在容器初始話的時(shí)候執(zhí)行這個(gè)方法
下面展示下web工程下每天00:00執(zhí)行任務(wù)的代碼:
@Component public class SystemInitListener implements ApplicationListener<ContextRefreshedEvent> { @Override public void onApplicationEvent(ContextRefreshedEvent event) { //創(chuàng)建定時(shí)器 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的幾種定時(shí)器小結(jié)
總結(jié)一下我使用過(guò)的4種類型的定時(shí)器:@Scheduled注解、quartz、new Timer().schedule、使用線程控制。
1.@Scheduled注解
@Scheduled注解是最簡(jiǎn)單的方式,只需要啟用定時(shí)器,在方法上添加注解即可。
在spring配置中加入:
<!-- 啟用注解定時(shí)器 --> <task:annotation-driven />
在要具體的方法上加入注解@Scheduled
@Scheduled(cron = "0 0 * * * ? ") public void myTask(){ //定時(shí)任務(wù)...... }
2.quartz
quartz使用的是可配置的方式,將所有的定時(shí)器都配置再一個(gè)xml文件里面。
步驟如下:
1.創(chuàng)建一個(gè)spring的配置文件:spring-quartz.xml
2.定義工作任務(wù)的job
3.定義觸發(fā)器Trigger并與job綁定
4.定義調(diào)度器,并將Trigger注冊(cè)到scheduler
<bean id="myTask" class="cn.coolwind.MyTask"/> <!-- 1.定義工作任務(wù)job --> <bean id="testJob" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean"> <!-- 定時(shí)器的類 --> <property name="targetObject" ref="myTask"></property> <!-- 需要定時(shí)執(zhí)行的方法 --> <property name="targetMethod" value="test"></property> <property name="concurrent" value="false"></property> </bean> <!-- 2.定義觸發(fā)器Trigger并與Job綁定 --> <bean id="testJobTrigger" class="org.springframework.scheduling.quartz.CronTriggerFactoryBean"> <property name="jobDetail" ref="testJob"/> <!-- 根據(jù)需要設(shè)置定時(shí)執(zhí)行的時(shí)間 --> <property name="cronExpression" value="0 0/5 * * * ?" /> </bean> <!-- 3.定義調(diào)度器,并將trigger注冊(cè)進(jìn)去 --> <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個(gè)參數(shù):
schedule(TimerTask task, long delay, long period)
第一個(gè)為定時(shí)任務(wù),根據(jù)業(yè)務(wù)需要重寫TimerTask的run方法即可;
第二個(gè)為延時(shí)啟動(dòng),單位毫秒;
第三個(gè)位多久運(yùn)行一次,單位毫秒;
new Timer().schedule(new TimerTask() { @Override public void run() { try { //do Something } catch (Exception e) { e.printStackTrace(); } } },0,5L * 60 * 1000);
4.使用線程控制
使用線程來(lái)控制就更靈活一些,可以根據(jù)自己的需要判斷什么時(shí)候運(yùn)行,什么時(shí)候停止,這需要對(duì)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(); } } } }); } }
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
springboot整合kaptcha驗(yàn)證碼的示例代碼
kaptcha是一個(gè)很有用的驗(yàn)證碼生成工具,本篇文章主要介紹了springboot整合kaptcha驗(yàn)證碼的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-06-06Java多線程之Callable接口的實(shí)現(xiàn)
這篇文章主要介紹了Java多線程之Callable接口的實(shí)現(xiàn),Callable和Runnbale一樣代表著任務(wù),區(qū)別在于Callable有返回值并且可以拋出異常。感興趣的小伙伴們可以參考一下2018-08-08Java連接Oracle數(shù)據(jù)庫(kù)并查詢
這篇文章主要介紹了Java連接Oracle數(shù)據(jù)庫(kù)并查詢的相關(guān)資料,需要的朋友可以參考下2017-04-04Java 在PPT中添加文本和圖片超鏈接的實(shí)現(xiàn)方法
這篇文章主要介紹了Java 在PPT中添加文本和圖片超鏈接的實(shí)現(xiàn)方法,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-05-05Spring中的ImportBeanDefinitionRegistrar接口詳解
這篇文章主要介紹了Spring中的ImportBeanDefinitionRegistrar接口詳解,ImportBeanDefinitionRegistrar接口是也是spring的擴(kuò)展點(diǎn)之一,它可以支持我們自己寫的代碼封裝成BeanDefinition對(duì)象,注冊(cè)到Spring容器中,功能類似于注解@Service @Component,需要的朋友可以參考下2023-09-09