调用Mybatis plus中的saveBatch方法报找不到表的问题

1.问题现象

在用Mybatis plus开发的项目中,用自带的API批量保存的方法saveBatch操作时,发现报没有找到表的错误。

错误日志截图如下:

表实际是存在的,且发现其他的方法都没有问题,包括save、update等单个的方法,都是正常的。百思不得其解,配置和代码都没有问题。

下面是该数据源对应的配置文件。

dataSourceConfig中配置的mapperLocations的位置:

其apollo中配置的value值为:classpath*:/mapper/*.xml

原来的maper.xml路径:

2.问题分析

在项目中用了2个数据源,上面是其中的一个数据源。

另一个数据源的mapper.xml配置的路径是:

其apollo中配置的value值也为:classpath*:/mapper/*.xml

本来是想保存数据到A数据库,结果是执行的是另一个数据库B,其实是因为 SqlSessionFactory 错误导致的。

项目中有2个数据源,分别用的不同的 SqlSessionFactory。

经分析跟踪代码后发现执行saveBatch后最终调用到了这个方法:

java 复制代码
 public static boolean executeBatch(Class<?> entityClass, Log log, Consumer<SqlSession> consumer) {
        SqlSessionFactory sqlSessionFactory = sqlSessionFactory(entityClass);
        SqlSessionHolder sqlSessionHolder = (SqlSessionHolder) TransactionSynchronizationManager.getResource(sqlSessionFactory);
        boolean transaction = TransactionSynchronizationManager.isSynchronizationActive();
        if (sqlSessionHolder != null) {
            SqlSession sqlSession = sqlSessionHolder.getSqlSession();
            //原生无法支持执行器切换,当存在批量操作时,会嵌套两个session的,优先commit上一个session
            //按道理来说,这里的值应该一直为false。
            sqlSession.commit(!transaction);
        }
        SqlSession sqlSession = sqlSessionFactory.openSession(ExecutorType.BATCH);
        if (!transaction) {
            log.warn("SqlSession [" + sqlSession + "] was not registered for synchronization because DataSource is not transactional");
        }

com.baomidou.mybatisplus.core.metadata.TableInfoHelper 中 initTableInfo方法会将每个实体类与对应的数据库配置保存到缓存:TABLE_INFO_CACHE

也就是,我们在创建 SqlSessionFactory 时候设置的 setMapperLocations, 设置路径下的所有mapper.xml 对应的实体都会保存对应的数据库配置。

因此,我们需要将不同的 SqlSessionFactory 配置,用不同的 mapper 目录来扫描。不同数据源的操作,放在各自的 mapper 子目录下,作区分。

上面我们定义2个SqlSessionFactory中配置的mapper文件路径是一样的。

由于 classpath*:mapper/**/*Mapper.xml 路径一样,导致初始化实体类和数据库配置对应关系,被覆盖的现象。

即同样的 mapper.xml 文件中被不同的 SqlSessionFactory 扫描了两次,导致mapper.xml中的实体类信息只有一种SqlSessionFactory信息。

3.解决办法

需要将其中的一个数据源对应的mapper文件映射的路径改一下,改成和另一个数据源配置的路径不同,就可以了。

修改后的路径:

相关推荐
2501_91676654几秒前
【Java】HashMap集合实现类
java·开发语言
海棠AI实验室4 分钟前
Python 学习路线图:从 0 到 1 的最短闭环
开发语言·python·学习
不会聊天真君6476 分钟前
设计模式、线程状态、上下文切换、线程安全(JAVA并发第二期)
java
Macbethad7 分钟前
技术报告:加密算法实现与性能优化研究
开发语言
玄同7657 分钟前
Python 函数:LLM 通用逻辑的封装与复用
开发语言·人工智能·python·深度学习·语言模型·自然语言处理
Swift社区8 分钟前
死锁:线程卡死不是偶然,而是设计问题
java·spring·maven
uup8 分钟前
防止短信验证码接口被盗刷问题
java
lkbhua莱克瓦2412 分钟前
基础-事务
开发语言·数据库·笔记·mysql·事务
xxxmine15 分钟前
ConcurrentHashMap 和 Hashtable 的区别详解
java·开发语言
凛_Lin~~15 分钟前
安卓 面试八股文整理(原理与性能篇)
android·java·面试·安卓