当前位置: 首页 > ops >正文

【Bug】定时任务中 Jpa Save 方法失效

【Bug】定时任务中 Jpa Save 方法失效

首先说一下问题,在定时任务中调用 jpa 的 save 方法没有效果,但是通过外界调用,比如 controller 中注入 service 来调用是可以的,真是巨巨巨离谱,我被折磨了好几天。

我这个问题的根源在于多数据源的配置上,所以如果你的项目中没有使用多数据源,那接下来的内容应该对你没什么帮助了~

一般来讲,在配置多数据源的时候,除了要继承 AbstractRoutingDataSource 类

package com.xsdl.content;import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;public class MultiDataSource extends AbstractRoutingDataSource {private static final ThreadLocal<String> keys = new ThreadLocal<String>();@Overrideprotected Object determineCurrentLookupKey() {return keys.get();}public static void setDataSource(String dataSource) {keys.set(dataSource);}public static String getDatasource() {return keys.get();}public static void clearDataSource() {keys.remove();}}

还会配置对应数据源的 datasource、jdbcTemplate 等

package com.xsdl.config;import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Description;
import org.springframework.context.annotation.Primary;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.PlatformTransactionManager;import javax.sql.DataSource;
import java.util.HashMap;
import java.util.Map;@Configuration
public class MultiDataSourceConfig {@Bean(name = "dataSourceAAA")@Primary@ConfigurationProperties("spring.aaa.datasource")public DataSource dataSourceAAA() {return DataSourceBuilder.create().build();}@Bean(name = "dataSourceBBB")@ConfigurationProperties("spring.bbb.datasource")public DataSource dataSourceBBB() {return DataSourceBuilder.create().build();}@Bean(name = "multiDataSource")public MultiDataSource multipleDataSource(@Qualifier("dataSourceAAA") DataSource dataSourceA,@Qualifier("dataSourceBBB") DataSource dataSourceB) {Map<Object, Object> map = new HashMap<>();map.put("dataSourceA", dataSourceA);map.put("dataSourceB", dataSourceB);MultiDataSource multipleDataSource = new MultiDataSource();multipleDataSource.setDefaultTargetDataSource(dataSourceA);multipleDataSource.setTargetDataSources(map);return multipleDataSource;}@Beanpublic JdbcTemplate jdbcTemplate(MultiDataSource multipleDataSource) {return new JdbcTemplate(multipleDataSource);}@Bean@Primary@Description("事务管理器")public PlatformTransactionManager transactionManager(@Qualifier("multiDataSource") MultiDataSource multiDataSource) {return new DataSourceTransactionManager(multiDataSource);}}

事情到现在是很正常的,启动项目也可以正常启动,正常在 service 的业务逻辑里调用对应的 dao 或者 repository 去保存数据也可以

突然有一天,我接到一个需求,有一个接收记录的表,我需要定时从中查询执行失败的,重试进行补偿,那我补偿完成后,当然要更新回去对吧,于是我写了一个类似下面这样的代码:

package com.xsdl.job;import com.xsdl.dao.UserDao;
import com.xsdl.pojo.User;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;import java.util.UUID;
import java.util.stream.IntStream;@Component("tsJob")
@Slf4j
public class TsJob {@Autowiredprivate UserDao userDao;@Scheduled(cron = "0 0/2 * * * ?")public void execute() {log.warn("定时任务开始执行");IntStream.rangeClosed(1, 10).forEach(i -> {User user = new User();user.setUuid(UUID.randomUUID().toString());user.setName("name" + i);userDao.save(user);});log.warn("定时任务执行结束");}}

具体逻辑不用关注,就是查数据库,然后修改了几个字段,重新 save 回去,这时候问题出现了,save 方法执行无效,数据库里数据没更新,我一路 debug,期间各种磕磕绊绊,就是找不到问题,反而找到了几个同病相怜的人:

[在spring的定时任务中使用 jpa 的save 方法失效](https://forum.springdoc.cn/t/topic/776)

no transaction is in progress

我一开始以为是事务的开启问题,包括但这样尝试:

    @Scheduled(cron = "0 0/2 * * * ?")public void execute() {log.warn("定时任务开始执行");IntStream.rangeClosed(1, 10).forEach(i -> {User user = new User();user.setUuid(UUID.randomUUID().toString());user.setName("name" + i);getThis.doSave(user);});log.warn("定时任务执行结束");}@Transactional(propagation = Propagation.New)public void doSave(User user) {userDao.save(user);}private TsJob getThis() {return ApplicationContextUtil.getBean("tsJob",TsJob.class);}

然而什么用都没有

最关键的是,这个方法如果在一个 controller 里手动调用是可以执行的,我觉得太离谱了,就想开个新项目复现一下,结果怎么都复现不出来,最后突然觉得区别也就是在多数据源上了,中间各种尝试,直到我把这个代码注释掉,终于可以了~

    @Bean@Primary@Description("事务管理器")public PlatformTransactionManager transactionManager(@Qualifier("multiDataSource") MultiDataSource multiDataSource) {return new DataSourceTransactionManager(multiDataSource);}

如果你的项目里也这样配置了 transactionManager ,可能就是这个问题~

http://www.xdnf.cn/news/9601.html

相关文章:

  • 今天遇到的bug
  • 基于大数据的个性化购房推荐系统设计与实现(源码+定制+开发)面向房产电商的智能购房推荐与数据可视化系统 基于Spark与Hive的房源数据挖掘与推荐系统设计
  • Blob文件导出:FileReader是否必需?✨
  • 由enctype-引出post与get的关系,最后深究至请求/响应报文
  • 2026-2030工业发展解读系列:PLM/PDM系统解析报告(预览版)
  • 自动化测试常见函数(下篇)
  • 单卡即可微调大模型!内存占用仅1/8
  • 【全因子组及排序】2022-1-23
  • 技术为器,服务为本:AI时代的客服价值重构
  • 高效推理引擎深度解析:vLLM 与 TGI 架构设计与性能实战指南
  • 在 Linux 中,目录权限,mkdir -m 选项,用法
  • 机器人--里程计
  • leetcode hot100刷题日记——24.回文链表
  • DOM和BOM的区别
  • pip国内镜像源配置
  • 【HW系列】—Log4j2、Fastjson、Shiro漏洞流量特征
  • Dify:详解 docker-compose.yaml配置文件
  • 解答:鲜羊奶真的能帮助青少年心理健康吗?
  • 框架漏洞(2)shiro
  • 数据结构- 10种常见树:二叉树、平衡二叉树、完全二叉树
  • 性能测试怎么做?方法、流程与核心要点解析
  • 直接偏好优化(DPO):用更简单的方法让 AI 更符合人类偏好
  • 文件上传之图片马文件头绕过(upload-labs通关笔记-第14关)
  • 时序预测力作PatchMixer论文理解
  • 5.28本日总结
  • [蓝帽杯 2022 初赛]网站取证_2
  • 第十五届蓝桥杯大赛软件赛国赛Python 大学 C 组试做【本期题单: 设置密码、栈】
  • Docker安装 | Spug
  • 【Linux】systemctl 和 sysctl 的区别
  • 常见的文件夹操作(附源码)