spring内部有一个task是Spring自带的一个设定时间自动任务调度
task使用的时候很方便,但是他能做的东西不如quartz那么的多!
可以使用注解和配置两种方式,配置的方式如下
引入Spring放在appcation.xml开头
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xmlns:aop="http://www.springframework.org/schema/aop"
- xmlns:context="http://www.springframework.org/schema/context"
- xmlns:task="http://www.springframework.org/schema/task"
- xsi:schemaLocation="
- http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
- http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
- http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
- http://www.springframework.org/schema/task http://www.springframework.org/schema/task/spring-task-4.0.xsd"
- default-lazy-init="true">
<!-- 注册bean -->
- <bean id="voiceFileClearJob" class="com.zjr.modules.boss.job.VoiceFileClearJob" />
- <bean id="versionListenJob" class="com.zjr.modules.boss.job.VersionListenJob" />
- <bean id="statJob" class="com.zjr.modules.opstat.job.StatJob" />
<!-- 开启任务调度 -->
- <task:scheduled-tasks>
- <task:scheduled ref="voiceFileClearJob" method="execute" initial-delay="5000" fixed-delay="3600000"/>
- <task:scheduled ref="versionListenJob" method="execute" initial-delay="5000" fixed-delay="5000"/>
- <task:scheduled ref="statJob" method="statLgj" cron="0 59 23 * * ?"/>
- <task:scheduled ref="statJob" method="statBadNameAndQQ" cron="23 28 20 * * ?"/>
- </task:scheduled-tasks>
第一个任务表示程序启动5s后调用voiceFileClearJob类中的execute方法,然后每隔一个小时再调用execute一次
第三个任务表示每天的23点59分调用statJob类中的statLgj方法
ref是工作类
method是工作类中要执行的方法
initial-delay是任务第一次被调用前的延时,单位毫秒
fixed-delay是上一个调用完成后再次调用的延时
fixed-rate是上一个调用开始后再次调用的延时(不用等待上一次调用完成)
cron是表达式,表示在什么时候进行任务调度。
以下为上述versionListenJob类的代码,实现的功能是监控某个文件夹,如果该文件夹里面的内容有任何改动,就重新生成一个txt文件,文件记录的是该文件夹所有文件的相关信息。
- package com.zjr.modules.boss.job;
-
- import ...
-
- public class VersionListenJob {
-
- Logger logger = LoggerFactory.getLogger(VersionListenJob.class);
-
- private static final String GMBOSS_DIR = "/data/jweb_static/jweb_wb_mgmt_beta/gmboss/";
- private static final String GMBOSS_VERSION_TXT_FILE = "/data/jweb_static/jweb_wb_mgmt_beta/txt/version.txt";
- private static final String DOWNLOAD_URL = "http://beta.wbmgmt.youzijie.com/gmboss/";
-
- private Map<String, String> versionInfoMap = new HashMap<>();
- private File dir = new File(GMBOSS_DIR);
- private File versionTxt = new File(GMBOSS_VERSION_TXT_FILE);
-
- public VersionListenJob() {
- try {
- init();
- } catch (IOException e) {
- logger.error("error occurs during VersionListenJob", e);
- }
- }
-
- private void init() throws IOException {
-
- if (!versionTxt.exists())
- versionTxt.createNewFile();
-
- for (String line : FileUtils.readLines(versionTxt)) {
- String[] array = StringUtils.split(line, "|");
- if (array.length == 5)
- versionInfoMap.put(array[0], array[4]);
- }
- }
-
- public void execute() throws IOException {
-
- if (EnvironmentUtil.isLocal() || !dir.exists() || !dir.isDirectory()) {
- return;
- }
-
-
- Boolean needUpdate = false;
- List<File> allFiles = getFiles(dir.getAbsolutePath());
-
- for (File file : allFiles) {
- String filePath = file.getPath();
- long lastModifiedTime = file.lastModified();
- if (!StringUtils.equals(lastModifiedTime + "", versionInfoMap.get(filePath))) {
- needUpdate = true;
- break;
- }
- }
-
- if (needUpdate) {
- List<String> list = new ArrayList<>();
- for (File file : allFiles) {
- list.add(file.getAbsolutePath().replace(GMBOSS_DIR,"/") + "|"
- + md5(file) + "|"
- + file.getPath().replace(GMBOSS_DIR,DOWNLOAD_URL) + "|"
- + file.length() + "|"
- + file.lastModified());
- }
- FileUtils.writeLines(versionTxt,"UTF-8", list, IOUtils.LINE_SEPARATOR_WINDOWS,false);
- logger.info("VersionListenJob: Gmboss has been updated");
- }
- }
-
-
- public List<File> getFiles(String filePath) throws IOException {
-
- List<File> allFiles = new ArrayList<>();
-
- File root = new File(filePath);
- File files[] = root.listFiles();
- if (files != null && files.length != 0) {
- for (File file : files) {
- if (file.isDirectory()) {
- allFiles.addAll(getFiles(file.getAbsolutePath()));
- } else {
- allFiles.add(file);
- }
- }
- }
- return allFiles;
- }
-
- public String md5(File f) {
- MessageDigest md = null;
- try {
- md = MessageDigest.getInstance("MD5");
- } catch (NoSuchAlgorithmException ne) {
- ne.printStackTrace();
- }
- if (md == null)
- return null;
- FileInputStream fis = null;
- try {
- fis = new FileInputStream(f);
- byte[] buffer = new byte[8192];
- int length;
- while ((length = fis.read(buffer)) != -1) {
- md.update(buffer, 0, length);
- }
- return new String(Hex.encodeHex(md.digest())).toUpperCase();
- } catch (Exception e) {
- logger.error("error occurs during md5 file", e);
- return null;
- } finally {
- try {
- if (fis != null)
- fis.close();
- } catch (IOException e) {
- logger.error("error occurs during md5 file", e);
- }
- }
- }
- }
格式: [秒] [分] [小时] [日] [月] [周] [年]
序号 | 说明 | 是否必填 | 允许填写的值 | 允许的通配符 |
1 | 秒 | 是 | 0-59 | , - * / |
2 | 分 | 是 | 0-59 | , - * / |
3 | 小时 | 是 | 0-23 | , - * / |
4 | 日 | 是 | 1-31 | , - * ? / L W |
5 | 月 | 是 | 1-12 or JAN-DEC | , - * / |
6 | 周 | 是 | 1-7 or SUN-SAT | , - * ? / L # |
7 | 年 | 否 | empty 或 1970-2099 | , - * / |
通配符说明:
* 表示所有值. 例如:在分的字段上设置 "*",表示每一分钟都会触发。
? 表示不指定值。使用的场景为不需要关心当前设置这个字段的值。例如:要在每月的10号触发一个操作,但不关心是周几,所以需要周位置的那个字段设置为"?" 具体设置为 0 0 0 10 * ?
- 表示区间。例如 在小时上设置 "10-12",表示 10,11,12点都会触发。
, 表示指定多个值,例如在周字段上设置 "MON,WED,FRI" 表示周一,周三和周五触发
/ 用于递增触发。如在秒上面设置"5/15" 表示从5秒开始,每增15秒触发(5,20,35,50)。 在月字段上设置'1/3'所示每月1号开始,每隔三天触发一次。
L 表示最后的意思。在日字段设置上,表示当月的最后一天(依据当前月份,如果是二月还会依据是否是润年[leap]), 在周字段上表示星期六,相当于"7"或"SAT"。如果在"L"前加上数字,则表示该数据的最后一个。例如在周字段上设置"6L"这样的格式,则表示“本 月最后一个星期五"
W 表示离指定日期的最近那个工作日(周一至周五). 例如在日字段上设置"15W",表示离每月15号最近的那个工作日触发。如果15号正好是周六,则找最近的周五(14号)触发, 如果15号是周未,则找最近的下周一(16号)触发.如果15号正好在工作日(周一至周五),则就在该天触发。如果指定格式为 "1W",它则表示每月1号往后最近的工作日触发。如果1号正是周六,则将在3号下周一触发。(注,"W"前只能设置具体的数字,不允许区间"-").
小提示 | 'L'和 'W'可以一组合使用。如果在日字段上设置"LW",则表示在本月的最后一个工作日触发(一般指发工资 ) |
# 序号(表示每月的第几个周几),例如在周字段上设置"6#3"表示在每月的第三个周六.注意如果指定"#5",正好第五周没有周六,则不会触发该配置(用在母亲节和父亲节再合适不过了)
小提示 | 周字段的设置,若使用英文字母是不区分大小写的 MON 与mon相同. |
常用示例:
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分每分触发 |
0 0/5 14 * * ? | 每天下午的 2点到2点59分(整点开始,每隔5分触发) |
0 0/5 14,18 * * ? | 每天下午的 2点到2点59分(整点开始,每隔5分触发) 每天下午的 18点到18点59分(整点开始,每隔5分触发) |
0 0-5 14 * * ? | 每天下午的 2点到2点05分每分触发 |
0 10,44 14 ? 3 WED | 3月分每周三下午的 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 | 每月的第三周的星期五开始触发 |
0 0 12 1/5 * ? | 每月的第一个中午开始每隔5天触发一次 |
0 11 11 11 11 ? | 每年的11月11号 11点11分触发(光棍节) |