• 源码分析Mybatis拦截器(Interceptor)拦截saveBatch()获取不到实体id的原因


    1.背景

    由于业务需求想在Mybatis拦截器层面获取insert后实体id去做相关业务。但是发现执行saveBatch()方法时,获取参数实体的时候,拿不到自增id。但是save()方法可以。

    save方法之所以可以是因为:

    MybatisPlus的BaseMapper执行insert方法后实体带自增id的原因是,在数据库表中设置了主键自增属性。当插入一条新的记录时,数据库会自动为这条记录生成一个唯一的自增id,并将这个id赋值给实体类中的主键属性。因此,当你查询这条记录时,实体类中的主键属性就会带有自增id。

    save()和saveBatch()的基本原理:

    MybatisPlus是MyBatis的增强工具,它在MyBatis的基础上只做增强不做改变,为简化开发、提高效率而生。在具体操作上,MybatisPlus提供了save()和saveBatch()两种方法用于数据的保存。

    save()方法是通过调用BaseMapper封装的单条保存数据的insert()方法实现的。也就是说,当我们调用save()方法时,会直接将数据插入到数据库中。

    而saveBatch()方法则是用于批量插入数据。在使用时,需要传入一个列表集合以及一个默认的批量插入数量(默认为1000)。在内部实现上,MybatisPlus会遍历这个列表集合,并依次将数据插入到数据库中。此外,需要注意的是,虽然saveBatch()方法可以实现批量插入,但根据实际测试和观察,批量插入的效率可能会受到一定影响。因此,在使用saveBatch()方法进行批量插入时,可能需要考虑优化方案以提高插入效率。

    拦截器代码(简化版):

    1. @Intercepts({@Signature(type = Executor.class, method = "update", args = {MappedStatement.class, Object.class})})
    2. public class TodoInterceptor implements Interceptor {
    3. @Override
    4. public Object intercept(Invocation invocation) throws Throwable {
    5. Object[] args = invocation.getArgs();
    6. Object parameter = args[1];
    7. // 调用原始方法
    8. Object result = invocation.proceed();
    9. if (parameter instanceof Enquiry) {
    10. Enquiry enquiry = (Enquiry) parameter;
    11. }
    12. return result;
    13. }

    2.现象

    我分别执行save和saveBatch方法,附带截图:

    save():

    saveBatch():

    如图所示,saveBatch()方法的实体在拦截器里是获取不到自增id的。特别说明下,debug你会发现,MybatisPlus执行saveBatch()方法会遍历这个列表集合,并依次将数据插入到数据库中。因为拦截器会执行次数是和实体列表的数量是一致。

    3.源码分析原因

    第一步:

    ServiceImpl类

    1. public boolean saveBatch(Collection entityList, int batchSize) {
    2. String sqlStatement = this.getSqlStatement(SqlMethod.INSERT_ONE);
    3. return this.executeBatch(entityList, batchSize, (sqlSession, entity) -> {
    4. //这里是拼装SQL,即每个实体类都是一个insert into 的sql
    5. sqlSession.insert(sqlStatement, entity);
    6. });
    7. }

    第二步:

    SqlHelper类
    1. public static boolean executeBatch(Class entityClass, Log log, Collection list, int batchSize, BiConsumer consumer) {
    2. Assert.isFalse(batchSize < 1, "batchSize must not be less than one", new Object[0]);
    3. return !CollectionUtils.isEmpty(list) && executeBatch(entityClass, log, (sqlSession) -> {
    4. int size = list.size();
    5. int idxLimit = Math.min(batchSize, size);
    6. int i = 1;
    7. //从这里for可以看出,每实体就是一个insert语句
    8. for(Iterator var7 = list.iterator(); var7.hasNext(); ++i) {
    9. E element = var7.next();
    10. //这里就是与数据库配置,执行了sql,同时拦截器也是在这个方法里面执行的
    11. consumer.accept(sqlSession, element);
    12. if (i == idxLimit) {
    13. //这里很重要,刷新缓存,执行这里后可以拿到插入后的id。
    14. //所以,大家想想,上面accept方法拿不到id的原因就是如此了。
    15. sqlSession.flushStatements();
    16. idxLimit = Math.min(idxLimit + batchSize, size);
    17. }
    18. }
    19. });
    20. }

    展示debug结果:

    大家可以看到,执行了sqlSession.flushStatements()方法后就出现了id。所以也是大家发现在controller层,service层等执行saveBatch方法能拿到id,在拦截器却不可以,原因就在这里

  • 相关阅读:
    七彩云南文化旅游网站的设计
    UV 裂解的生物素-PEG2-叠氮|CAS:1192802-98-4生物素接头
    mysql数据库基础:视图、变量
    Java-使用sqlSessionTemplate实现批量更新-模拟mybatis 动态sql
    医疗器械设计时需要注意的事项
    Python与MySQL交互
    【实践篇】Redis最强Java客户端(一)之Redisson入门介绍
    javaee ssm框架整合例子 ssm例子,需要哪些依赖,配置文件如何配置
    c语言-实用调试技巧
    近年来国内室内定位领域硕士论文选题的现状与趋势
  • 原文地址:https://blog.csdn.net/taipoucha5799/article/details/134496738