Spring中定时任务@Scheduled的一点小小研究
2021-07-04 02:11
标签:代码 iat 技术 说明 block 冲突 png upd 服务端 最近做一个公众号项目,微信公众号会要求服务端找微信请求一个access_token,获取的过程: access_token是公众号的全局唯一接口调用凭据,公众号调用各接口时都需使用access_token。开发者需要进行妥善保存。access_token的存储至少要保留512个字符空间。access_token的有效期目前为2个小时,需定时刷新,重复获取将导致上次获取的access_token失效。 公众平台的API调用所需的access_token的使用及生成方式说明: 1、建议公众号开发者使用中控服务器统一获取和刷新Access_token,其他业务逻辑服务器所使用的access_token均来自于该中控服务器,不应该各自去刷新,否则容易造成冲突,导致access_token覆盖而影响业务; 2、目前Access_token的有效期通过返回的expire_in来传达,目前是7200秒之内的值。中控服务器需要根据这个有效时间提前去刷新新access_token。在刷新过程中,中控服务器可对外继续输出的老access_token,此时公众平台后台会保证在5分钟内,新老access_token都可用,这保证了第三方业务的平滑过渡; 3、Access_token的有效时间可能会在未来有调整,所以中控服务器不仅需要内部定时主动刷新,还需要提供被动刷新access_token的接口,这样便于业务服务器在API调用获知access_token已超时的情况下,可以触发access_token的刷新流程。 所以解决方案就是做一个定时任务,服务启动的时候获取一次token,之后每隔两小时自动获取一次。这用Spring的@Scheduled注解很容易实现: 这里使用fixedRate来指定任务执行的时间间隔(毫秒记)。 然后因为现在项目只有一个服务号,没有申请测试用的服务号,本地调试也会去获取access_token ,结果导致服务端的生产环境的token失效,必须重启服务器的tomcat来刷新token。显然这种做法是非常愚蠢的,目前还没正式投入使用,投入使用后这显然不能接受。 解决方案就是,统一一个接口来向服务端获取token,当token失效时,执行一次updateToken的方法,以刷新token。 ===================================================================手动分割线============================================================ 然后我就比较好奇一个事,那就是,当加了@Scheduled(fixedRate=1000*60*59*2)的方法被其他方法调用的时候,定时任务会有变化吗?比如,我调用一次,那么任务是还是按照以前预定的时间跑呢,还是会在我调用之后的2小时(预约的间隔时间)才会再跑?百度Google找了找,包括官方文档也没给个说法。为这个也不想去看源代码,还是写个demo验证下算了。 首先创建一个Spring Boot的项目,引入web模块(使用rest请求来手动调用定时任务) 项目结构: Service接口: Service实现(每隔10秒钟在控制台打印一次Hello!): Controller: 然后在SpringBoot的启动类加入定时任务的注解@EnableScheduling 启动应用,可以看到控制台可以定时打印Hello和时间: 在Postman给服务端发一个请求,手动调用定时任务: get http://localhost:8080/demo/sayHello 结果: 可以看到定时任务没有受到调用的影响,还是按照自己的节奏10秒钟一次~~ (而且,还会发现Spring会在第一次收到Http请求的时候才会初始化dispatcherServlet,而不是服务启动的时候就初始化) Spring中定时任务@Scheduled的一点小小研究 标签:代码 iat 技术 说明 block 冲突 png upd 服务端 原文地址:https://www.cnblogs.com/xtf2009/p/9614496.html
@Override
@Scheduled(fixedRate=1000*60*59*2)
public void updateToken() {
try{
for(AppConfig appConfig : SysConfig.AppLists){
Token token = TokenAPI.token(appConfig.wxAppId,appConfig.wxAppSecret);
appConfig.access_token = token.getAccess_token();
Ticket ticket = TicketAPI.ticketGetticket(token.getAccess_token());
appConfig.ticket = ticket.getTicket();
logger.info("access_token:" + token.getAccess_token()+",js_api_ticket:"+ticket.getTicket());
}
}catch (Exception e){
logger.info("刷新access_token出错");
}
}
dependency>
groupId>org.springframework.bootgroupId>
artifactId>spring-boot-starter-webartifactId>
dependency>
package com.xiatingfei.schedule.demo.service;
public interface TaskService {
void SayHelloTask();
}
package com.xiatingfei.schedule.demo.service.impl;
import com.xiatingfei.schedule.demo.service.TaskService;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Service;
import java.text.SimpleDateFormat;
import java.util.Date;
@Service
public class TaskServiceImpl implements TaskService {
private static final SimpleDateFormat dateFormat = new SimpleDateFormat("HH:mm:ss");
@Override
@Scheduled(fixedRate = 10000)
public void SayHelloTask() {
System.out.println("Hello! Now is " + dateFormat.format(new Date()));
}
}
package com.xiatingfei.schedule.demo.controller;
import com.xiatingfei.schedule.demo.service.TaskService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/demo")
public class TriggerController {
@Autowired
TaskService taskService;
@RequestMapping("/sayHello")
public void TriggerTask(){
taskService.SayHelloTask();
}
}
package com.xiatingfei.schedule.demo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.scheduling.annotation.EnableScheduling;
@SpringBootApplication
@EnableScheduling
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
上一篇:常用的python内置方法
文章标题:Spring中定时任务@Scheduled的一点小小研究
文章链接:http://soscw.com/essay/101533.html