Spring学习—Spring中定时器实现
Spring学习—Spring中定时器实现
在一些工作需要使用到定时器,Spring很好的集成了定时器的功能!
在Spring 中使用Quartz,本文介绍Spring3.0以后自主开发的定时任务工具,spring task,可以将它比作一个轻量级的Quartz,而且使用起来很简单,除spring相关的包外不需要额外的包,
下面介绍两种方式实现Spring定时器功能,一种是基于xml配置方式,一种是基于注解的方式,大家根据自己的项目选择适合自己的。
文章目录
一:基于xml配置的方式
1:编写普通的pojo 类
package com.aflyun.web.task;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Service;
@Component
//@Service 都可以
public class TaskCool {
/**
* 第一个定时器测试方法
*/
public void testJob(){
System.out.println("test first taskJob .... ");
}
}
2:配置xml文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:task="http://www.springframework.org/schema/task"
xmlns="http://www.springframework.org/schema/beans"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/task
http://www.springframework.org/schema/task/spring-task.xsd">
<context:component-scan base-package="com.aflyun.web" />
<aop:aspectj-autoproxy proxy-target-class="true" />
<context:annotation-config />
<!-- 在applicationContext.xml中进行配置,使用定时器
ref : pojo类的名称
method : 调用的方式名称
cron : cronExpression表达式
cron="0/5 * * * * ?" //表示五秒钟执行一次
-->
<task:scheduled-tasks>
<task:scheduled ref="taskCool" method="testJob" cron="0/5 * * * * ?"/>
</task:scheduled-tasks>
</beans>
注:上面主要的配置文件中一定要加入task的命名空间和schema。
上面 ref=”taskCool”,默认为这个TaskCool 类的首字母小写的值,
若需要修改可以在@Component里面进行修改 ,例如下面
@Component(“taskCoolJob”) 则此时 ref=”taskCoolJon”。
到此基于xml配置完成,运行则可以看到效果!
二:基于注解方式
使用注解方式不需要再每写一个任务类还要在xml文件中配置下,方便了很多。使用Spring的@Scheduled,下面先看一注解@Scheduled在源文件中的定义:
@Target({java.lang.annotation.ElementType.METHOD, java.lang.annotation.ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Scheduled
{
public abstract String cron();
public abstract long fixedDelay();
public abstract long fixedRate();
}
cron:表示指定cron表达式。(cron类型表示 是指定时间触发器触发任务执行!)
- fixedDelay:表示从上一个任务完成开始到下一个任务开始的间隔,单位是毫秒。
- fixedRate:表示从上一个任务开始到下一个任务开始的间隔,单位是毫秒。
下面进行一下具体的配置过程:
1:编写pojo类
package com.tclshop.cms.center.web.task;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
@Component
public class WebTask {
// 每五秒执行一次
@Scheduled(cron = "0/5 * * * * ?")
public void TaskJob() {
System.out.println("test second annotation style ...");
}
}
2:配置xml文件
下面贴出相关的配置文件内容:
<!-- 开启这个配置,spring才能识别@Scheduled注解 -->
<task:annotation-driven scheduler="qbScheduler" mode="proxy"/>
<task:scheduler id="qbScheduler" pool-size="10"/>
注:理论上只需要加上这句配置就可以了,其他参数都不是必须的。
配置完成,运行就能看到效果!
总结:这种定时器的使用,不需要集成其他父类定时器,使用简单方便!功能也很强大!
附:cronExpression的配置说明
字段 | 允许值 | 允许的特殊字符 |
---|---|---|
秒 | 0-59 | , - * / |
分 | 0-59 | , - * / |
小时 | 0-23 | , - * / |
日期 | 1-31 | , - * ? / L W C |
月份 | 1-12 或者 JAN-DEC | , - * / |
星期 | 1-7 或者 SUN-SAT | , - * ? / L C # |
年(可选) | 留空, 1970-2099 | , - * / |
例子:
CRON表达式 | 含义 |
---|---|
“0 0 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:55分结束每5分钟一次触发 |
“0 0/5 14,18 * * ?” | 每天的下午2点至2:55和6点至6点55分两个时间段内每5分钟一次触发 |
“0 0-5 14 * * ?” | 每天14:00至14:05每分钟一次触发 |
“0 10,44 14 ? 3 WED” | 三月的每周三的14:10和14:44触发 |
“0 15 10 ? * MON-FRI” | 每个周一、周二、周三、周四、周五的10:15触发 |
三:Spring @Scheduled定时任务动态修改cron参数
Spring框架自3.0版本起,自带了任务调度功能,好比是一个轻量级的Quartz,而且使用起来也方便、简单,且不需要依赖其他的JAR包。秉承着Spring的一贯风格,Spring任务调度的实现同时支持注解配置和XML配置两种方式。
再来谈谈变态的项目需求:我们正在做一个智能数字电表的数据采集项目,项目最终会在多个工业园上线,每个工业园对电表数据的采集周期可以进行自定义,例如A工业园想每10分钟采集一次数据,B工业园想每15分钟采集一次数据。因为数据采集是个重复的周期性工作,那么就可以考虑使用Spring框架的定时任务功能了。
按正常来讲,修改定时任务的执行周期还不简单,把服务停下来,改下任务的cron参数,再重启服务就搞定了。但有没有一种可能,在不停服务的情况下,就可以动态的修改任务的cron参数呢?完全是有可能的!
直接看代码:
package com.tradeplatform.platform.user.job;
import java.math.BigDecimal;
import java.util.Date;
import java.util.List;
import java.util.Map;
import org.apache.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.Trigger;
import org.springframework.scheduling.TriggerContext;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.scheduling.annotation.SchedulingConfigurer;
import org.springframework.scheduling.config.ScheduledTaskRegistrar;
import org.springframework.scheduling.support.CronTrigger;
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional;
import com.tradeplatform.platform.trade.mapper.TradeConfigMapper;
import com.tradeplatform.platform.trade.service.TradeConfigService;
import com.tradeplatform.platform.user.service.ReleasedTaskService;
import com.tradeplatform.platform.user.service.UserAccountDetailService;
import com.tradeplatform.platform.user.service.UserAccountService;
import com.tradeplatform.trade.api.finan.utils.FinanceDetailRemark;
import com.tradeplatform.trade.api.finan.utils.FinanceDetailType;
import com.tradeplatform.trade.api.trade.utils.TradeConstants;
import com.tradeplatform.trade.api.user.entity.UserAccount;
/**
* 自动释放币
* <p>
* 创建人:pengyq <br>
* 创建时间:2018年5月31日 下午12:08:11 <br>
* <p>
* 修改人: <br>
* 修改时间: <br>
* 修改备注: <br>
* </p>
*/
@Component
@EnableScheduling
@Transactional
public class ReleasedTask implements SchedulingConfigurer{
private Logger log = Logger.getLogger(getClass());
@Autowired
private ReleasedTaskService taskService;
private TradeConfigMapper tradeConfigMapper;
@Autowired
private TradeConfigService configService;
@Autowired
private UserAccountService accountService;
@Autowired
private UserAccountDetailService accountDetailService;
/**
* 查询出符合条件的记录(新)
* <p>
* 创建人:lgq <br>
* 创建时间:2018年5月31日 下午1:55:01 <br>
* <p>
* 修改人: <br>
* 修改时间: <br>
* 修改备注: <br>
* </p>
*
* @return
*/
public List<Map<String, Object>> getNeedReleasedAccounts() {
return taskService.getNeedReleasedAccounts();
}
private String cron;
@Autowired
public ReleasedTask(TradeConfigMapper tradeConfigMapper) {
// 获取每几天释放一次参数(单位:天)
this.tradeConfigMapper = tradeConfigMapper;
String interval = tradeConfigMapper.getTimeInterval().getConfigValue();
this.cron = "0 0 3 */" + interval + " * ?";
//cron = "*/10 * * * * ?";
}
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
taskRegistrar.addTriggerTask(new Runnable() {
@Override
public void run() {
//是否执行自动释放
String autoRelease = configService.getConfig(TradeConstants.TRADE_CONFIG_AUTO_RELEASE);
if (autoRelease.equals("1")) {
// 任务逻辑
log.info("释放任务执行---------");
//查询出所有需要释放的钱包
List<Map<String, Object>> accounts = getNeedReleasedAccounts();
// 每次释放比例变量(全局)
BigDecimal percentageOfRelease = new BigDecimal(
configService.getConfig(TradeConstants.TRADE_CONFIG_PERCENTAGE_OF_RELEASE));
//待释放数量小于此参数时全部释放
BigDecimal releaseAll = new BigDecimal(
configService.getConfig(TradeConstants.TRADE_CONFIG_RELEASE_ALL));
//循环进行释放
for (Map<String, Object> map : accounts) {
//待释放总额
BigDecimal freezeAmount = new BigDecimal(map.get("freeze_amount").toString());
//本次循环释放金额
BigDecimal readyReleased = new BigDecimal("0");
//先判断待释放金额是否小于全部释放参数
if (releaseAll.compareTo(freezeAmount) == 1 || releaseAll.compareTo(freezeAmount) == 0) {
//直接一次性释放完
readyReleased = freezeAmount;
}else {
//本次释放金额
readyReleased = freezeAmount.multiply(percentageOfRelease);
}
// 更新对应的用户、币账户已释放余额、未释放余额
Long uId = Long.valueOf(map.get("user_id").toString());
Long cId = Long.valueOf(map.get("coin_id").toString());
// 账户已释放金额
int a = accountService.updateUnAmount(readyReleased.abs().negate(), uId, cId);
// 修改账户待释放金额
int b = accountService.updateAmount(readyReleased.abs().negate(), uId, cId);
if (a > 0 && b > 0) {
UserAccount account = accountService.getAccountByUserIdAndCoinId(uId, cId);
// 插入1条账户明细
accountDetailService.createPaymentDetail(uId, cId, account.getId(), account.getId(),
FinanceDetailType.TRADE_DETAIL_TYPE_INCOME, FinanceDetailRemark.RELEASED_COIN, readyReleased,
"释放", 0);
}
}
log.info("释放任务执行完---------");
}
}
}, new Trigger() {
@Override
public Date nextExecutionTime(TriggerContext triggerContext) {
// 任务触发,可修改任务的执行周期
CronTrigger trigger = new CronTrigger(cron);
Date nextExec = trigger.nextExecutionTime(triggerContext);
return nextExec;
}
});
}
}
上面的TradeConfigMapper 使用的构造注入,不然会拿不到值。
四:参考资料
作者:不敲代码的攻城狮
出处:https://www.cnblogs.com/leigq/
任何傻瓜都能写出计算机可以理解的代码。好的程序员能写出人能读懂的代码。