Java之旅--定时任务(Timer、Quartz、Spring、LinuxCron)
在Java中,实现定时任务有多种方式,本文介绍4种,Timer和TimerTask、Spring、QuartZ、Linux Cron。
以上4种实现定时任务的方式,Timer是最简单的,不需要任何框架,仅仅JDK就可以,缺点是仅仅是个时间间隔的定时器,调度简单;Spring和QuartZ都支持cron,功能都很强大,Spring的优点是稍微简单一点,QuartZ的优点是没有Spring也可使用;Linux Cron是个操作系统级别的定时任务,适用于所有操作系统支持的语言,缺点是精度只能到达分钟级别。
Timer和TimerTask
关于Timer定时器的实现原理,如果我们看过JDK源码,就会发现,是使用的Object.wait(timeout),来进行的线程阻塞,timeout是根据下次执行实际和当前实际之差来计算。实际上,这可以归结为一个多线程协作(协作都是在互斥下的协作)问题。
在java.util.concurrent中,有个ScheduledThreadPoolExecutor,也可以完全实现定时任务的功能。
而其他的框架,无非是功能的增强,特性更多,更好用,都是在基础的java之上的包装。
代码示例如下:
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
public class TimerTest extends TimerTask
{
    private Timer timer;
    public static void main(String[] args)
    {
        TimerTest timerTest= new TimerTest();
        timerTest.timer = new Timer();
        //立刻开始执行timerTest任务,只执行一次
        timerTest.timer.schedule(timerTest,new Date());
        //立刻开始执行timerTest任务,执行完本次任务后,隔2秒再执行一次
        //timerTest.timer.schedule(timerTest,new Date(),2000);
        //一秒钟后开始执行timerTest任务,只执行一次
        //timerTest.timer.schedule(timerTest,1000);
        //一秒钟后开始执行timerTest任务,执行完本次任务后,隔2秒再执行一次
        //timerTest.timer.schedule(timerTest,1000,2000);
        //立刻开始执行timerTest任务,每隔2秒执行一次
        //timerTest.timer.scheduleAtFixedRate(timerTest,new Date(),2000);
        //一秒钟后开始执行timerTest任务,每隔2秒执行一次
        //timerTest.timer.scheduleAtFixedRate(timerTest,1000,2000);  
        try
        {
            Thread.sleep(10000);
        } catch (InterruptedException e)
        {
            e.printStackTrace();
        }
        //结束任务执行,程序终止
        timerTest.timer.cancel();
        //结束任务执行,程序并不终止,因为线程是JVM级别的
        //timerTest.cancel();
    }
    @Override
    public void run()
    {
        System.out.println("Task is running!");
    }
}
使用spring @Scheduled注解执行定时任务
这种方式非常简单,却能使用cron完成和QuartZ一样的功能,值得推荐一下。
ApplicationContext.xml:
beans根节点增加内容:xmlns:task="http://www.springframework.org/schema/task"
beans根节点中,xsi:schemaLocation属性下增加:
http://www.springframework.org/schema/taskhttp://www.springframework.org/schema/task/spring-task-3.1.xsd
实现类:
@Component  //import org.springframework.stereotype.Component;
public class MyTestServiceImpl  implements IMyTestService {
      @Scheduled(cron="0/5 * *  * * ? ")   //每5秒执行一次
      @Override
      public void myTest(){
            System.out.println("进入测试");
      }
}
注意几点:
spring的@Scheduled注解 需要写在实现上;
定时器的任务方法不能有返回值;
实现类上要有组件的注解@Component,@Service,@Repository
QuartZ
QuartZ With Spring
applicationContext-schedule.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xmlns:context="http://www.springframework.org/schema/context"
 xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd">
 <bean id="quartzScheduler" class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
  <property name="triggers">
   <list>
    <!-- 启动的Trigger列表 -->
    <ref local="startThriftTrigger" />
   </list>
  </property>
  <property name="quartzProperties">
   <props>
    <prop key="org.quartz.threadPool.threadCount">5</prop>
   </props>
  </property>
  <!-- 启动时延期3秒开始任务 -->
  <property name="startupDelay" value="3" />
  <property name="autoStartup" value="${scheduler.autoStartup}" />
 </bean>
 <!-- 启动Thrift,立即启动 -->
 <bean id="startThriftTrigger" class="org.springframework.scheduling.quartz.SimpleTriggerBean">
  <property name="startDelay" value="0" />
        <property name="repeatInterval" value="1000" />
        <property name="repeatCount" value="0" />
        <property name="jobDetail" ref="startThriftTask" />
 </bean>
 <bean id="startThriftTask" class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
  <property name="targetObject" ref="startThrift" />
  <property name="targetMethod" value="execute" />
  <!-- 同一任务在前一次执行未完成而Trigger时间又到时是否并发开始新的执行, 默认为true. -->
  <property name="concurrent" value="true" />
 </bean>
</beans>
实现类
package xx.schedule;
@Component
public class StartThrift {
 /**
  * 调度入口
  */
 public void execute() {
  // to do something
 }
}
QuartZ No Spring
使用的QuartZ jar是1.8.5,如下:
<dependency>
    <groupId>org.quartz-scheduler</groupId>
    <artifactId>quartz</artifactId>
    <version>1.8.5</version>
</dependency>
调用类实现代码如下:
import org.quartz.CronExpression;
import org.quartz.CronTrigger;
import org.quartz.JobDetail;
import org.quartz.Scheduler;
import org.quartz.SchedulerException;
import org.quartz.SchedulerFactory;
import org.quartz.impl.StdSchedulerFactory;
public class InvokeStatSchedule {
    public void start() throws SchedulerException
    {
        SchedulerFactory schedulerFactory = new StdSchedulerFactory();
        Scheduler scheduler = schedulerFactory.getScheduler();
        //InvokeStatJob是实现了org.quartz.Job的类
        JobDetail jobDetail = new JobDetail("jobDetail", "jobDetailGroup", InvokeStatJob.class);
        CronTrigger cronTrigger = new CronTrigger("cronTrigger", "triggerGroup");
        try {
            CronExpression cexp = new CronExpression("0 0 * * * ?");
            cronTrigger.setCronExpression(cexp);
        } catch (Exception e) {
            e.printStackTrace();
        }
        scheduler.scheduleJob(jobDetail, cronTrigger);
        scheduler.start();
    }
}
定时任务类代码如下:
import org.quartz.Job;
import org.quartz.JobExecutionContext;
import org.quartz.JobExecutionException;
public class InvokeStatJob implements Job {
    @Override
    public void execute(JobExecutionContext arg0) throws JobExecutionException {
        //...要定时操作的内容
    }
}
Linux Cron
这其实也是一种非常普遍的实现定时任务的方式,实际是操作系统的定时任务。Linux Cron只能到达分钟级,到不了秒级别。
一般我们是设置定时执行一个sh脚本,在脚本里面写一些控制代码,例如,有如下的脚本:
3,33 * * * * /usr/local/log_parser/run_log_parser.sh &
run_log_parser.sh的内容大致如下:
#!/bin/sh
log_parser_dir=/usr/local/log_parser
tmp_file=/usr/local/run_parser_tmp.txt
parser_log=/usr/local/access_parser.log
tmpDir=/data/applogs/access_logs_bp
date >> "$parser_log"
if [! -f "$tmp_file"]; then
        echo '访问日志解析正在进行,尚未完成' >> "$parser_log"
        echo '' >> "$parser_log"
else
        echo '开始解析访问日志' >> "$parser_log"
        touch "$tmp_file"
        cd "$log_parser_dir"
        python access_log_parser.py >> "$parser_log"
        rm "$tmp_file"
        echo '解析访问日志完成' >> "$parser_log"
        echo '' >> "$parser_log"
        cd "$tmpDir"
        gzip gzip WEB0*
        mv *.gz gz/
        echo '压缩备份日志及移动到压缩目录成功' >> "$parser_log"
fi
from: https://www.kancloud.cn/digest/java-travel/159427												
											Java之旅--定时任务(Timer、Quartz、Spring、LinuxCron)的更多相关文章
- SpringBoot之旅 -- 定时任务两种(Spring Schedule 与 Quartz 整合 )实现
		
相关文章 Spring Boot 相关文章目录 前言 最近在项目中使用到定时任务,之前一直都是使用Quartz 来实现,最近看Spring 基础发现其实Spring 提供 Spring Schedul ...
 - Java定时任务调度工具Timer Quartz
		
-----------------------------------Timer PS:Timer是 jdk中的:Quartz是第三方的 PS:也即是返回执行任务时候的时间 ----------- ...
 - java定时任务实现的几种方式(Timer、Spring Task、Quartz)
		
Timer JDK自带的Timer类,允许调度一个TimerTask任务. Demo: /** * Timer测试类 */ public class TimerDemo { public static ...
 - atititt.java定时任务框架选型Spring Quartz 注解总结
		
atititt.java定时任务框架选型Spring Quartz 总结 1. .Spring Quartz (ati recomm) 1 2. Spring Quartz具体配置 2 2.1. 增 ...
 - spring定时任务之quartz
		
在Spring中,使用JDK的Timer类库来做任务调度功能不是很方便,关键它不可以象cron服务那样可以指定具体年.月.日.时和分的时间.你只能将时间通过换算成微秒后传给它.如任务是每天执行一次,则 ...
 - Spring整合quartz2.2.3总结,quartz动态定时任务,Quartz定时任务集群配置
		
Spring整合quartz2.2.3总结,quartz动态定时任务,Quartz定时任务集群配置 >>>>>>>>>>>>&g ...
 - java中实现定时任务 task 或quartz
		
转载大神的 https://www.cnblogs.com/hafiz/p/6159106.html https://www.cnblogs.com/luchangyou/p/6856725.html ...
 - [JAVA]定时任务之-Quartz使用篇
		
Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用.Quartz可以用来创建简单或为运行十个,百个, ...
 - Java基础--定时任务Timer
		
Java基础--定时任务Timer 一.Timer介绍 java.util.Timer java.util.TimerTask Timer是一个定时器类,通过该类可以为指定的定时任务进行配置.Time ...
 
随机推荐
- linux 检测远程端口是否打开
			
linux 检测远程端口是否打开 检测远程端口是否打开 常用telnet 110.101.101.101 80方式测试远程主机端口是否打开. 除此之外还可以使用: 方法1.nmap i ...
 - Codeforces Round #257 (Div. 1) D - Jzzhu and Numbers 容斥原理 + SOS dp
			
D - Jzzhu and Numbers 这个容斥没想出来... 我好菜啊.. f[ S ] 表示若干个数 & 的值 & S == S得 方案数, 然后用这个去容斥. 求f[ S ] ...
 - 基于jquery的简洁树形折叠菜单
			
先上效果图: 最小的ul就是一个最小的树枝,空间允许时可无限扩展.html如下: <div class="panel panel-default"> <div c ...
 - IEnumerable<T>
			
IEnumerable 饮水思源 <C#本质论> Overview 根据定义,.Net 的中集合,本质上是一个类,它最起码实现了IEnumeraable 或者非泛型的IEnumerable ...
 - flask run方法和run_simple
			
1.Flask提供的Web服务器不适合在生产环境中使用 2.run方法启动flask集成的服务器: 例: if __name__ == '__main__': app.run(debug=True) ...
 - QT学习笔记9:QTableWidget的用法总结
			
最近用QT中表格用的比较多,使用的是QTableWidget这个控件,总结一下QTableWidget的一些相关函数. 1.将表格变为禁止编辑: tableWidget->setEditTrig ...
 - MongoDB简单使用 —— 基本操作
			
本身MongoDB直接支持的是Bson文档,Bson文档在C#的官方驱动中对应的是BsonDocument类: var bsonDoc = new BsonDocument(){ [" ...
 - Maven具体解释之------maven版本号管理
			
本文同意转载,但请标明出处:http://blog.csdn.net/wanghantong/article/38424065, 版权全部 如今所说的maven版本号不同于SVN的版本号控制哦!!! ...
 - [译] Go 并发编程基础
			
原文:Fundamentals of concurrent programming 译者:youngsterxyf 本文是一篇并发编程方面的入门文章,以Go语言编写示例代码,内容涵盖: 运行期并发线程 ...
 - HDU4607(求树中的最长链)
			
题目:Park Visit 题意:给定一棵树,从树中的任意选一个顶点出发,遍历K个点的最短距离是多少?(每条边的长度为1) 解析:就是求树的最长链,假设求出的树的最长链所包含的点数为m,那么如果K&l ...