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。

推薦閱讀: