MybatisPlus字段自动填充失效,填充值为null的解决方案

 更新时间:2024年01月13日 09:23:36   作者:庸人冲  
这篇文章主要介绍了MybatisPlus字段自动填充失效,填充值为null的解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

问题描述

有一个实体类UserEntity 对其属性 UserEntity#createTime 字段注解了 @TableField(fill = FieldFill.INSERT)

image-20220521091544321

根据业务需要定义了 UserMapper#inserUser() 方法,参数注解了 @Param(“user”) 如下:

image-20220521091425353

当调用该方法时,无法给 createTime 字段自动填充值,报错信息如下:

### Cause: java.sql.SQLIntegrityConstraintViolationException: Column 'create_time' cannot be null<LF>; Column 'create_time' cannot be null; nested exception is java.sql.SQLIntegrityConstraintViolationException: Column 'create_time' cannot be null]

问题剖析

在检查了 MetaObjectHandler 实现类的重写的方法无误后,开始尝试跟踪 Mybatis-plus 的源码。

发现在 MybatisParameterHandler#process() 中完成了自动填充的功能,在自动填充前需要先获取 tableInfo 信息:

image-20220521093717931

而这个 TableInfoHelper.getTableInfo() 方法只有当传入的 Class 对象是实体类对象时才能获取到 tableInfo :

image-20220521094104208

那么问题来了,我的参数确实是实体类,但是为什么获取不到 TableInfo 呢?

因为 MybatisParameterHandler#process() 方法的 parameter 参数在调用我自定义的方法时,传入的是一个 Map ,这个 Mapkey 是一个字符串表示,而 value 是我自定义方法的参数实例,可以看到下图中我的 Map 有两个 Entry 一个 key=user 一个 key=param1

image-20220521094353846

而最为重要的是,在这个 process() 方法中,如果传入的是一个 Map,Mybatis-plus 会从其中取 key="et" 的值,这就是问题的原因所在!!!

image-20220521094759755

而传入的这个 Map 不存在 key="et" 的映射关系。因此两个 TableInfoHelper.getTableInfo() 方法都进不去,所以也就不会进行自动填充。

image-20220521095247096

那么如何建立 "et" -> entity 的映射关系呢?我们Map中原本的两个的映射关系又是从哪里来的?

根据方法的调用链,一直回退到 Mybatis 框架中的 MapperMethod#execute() 方法中的一行代码:

image-20220521095621950

上面的 convertArgsToSqlCommandParam() 方法就是通过我们方法的实际参数 args 转换为执行 sql 语句需要的参数格式,而返回值 param 就是之前传入的那个 map

我们跟踪该方法的调用链,发现最终调用了 ParamNameResolver#getNamedParams() 方法,该方法有3个分支,决定了我们最终得到参数是怎样的,源码如下:

  public Object getNamedParams(Object[] args) {
    final int paramCount = names.size();
     // 1. 方法是空参时直接返回
    if (args == null || paramCount == 0) {  
      return null;
    } else if (!hasParamAnnotation && paramCount == 1) { 
       // 2. 方法的参数没有注解 @Param 并且只有一个参数时,直接返回这个参数实例
      Object value = args[names.firstKey()];
      return wrapToMapIfCollection(value, useActualParamName ? names.get(0) : null);
    } else {
        // 3. 否则,就建立映射关系(要么注解了 @Param,要么就是多个参数)
      final Map<String, Object> param = new ParamMap<>();
      int i = 0;
        // 遍历  names 中每一个 entry, 这个 names 是一个 sortedMap,该 Map 会保存方法参数的索引 -> 参数名称的映射关系,如果参数注解了 @Param,则值时 @Param("xxx") 中的 xxx,如果没有注解 @Param 则值也为参数索引,例如:
        // aMethod(@Param("M") int a, @Param("N") int b) -> {{0, "M"}, {1, "N"}}
		// aMethod(int a, int b) -> {{0, "0"}, {1, "1"}}
      for (Map.Entry<Integer, String> entry : names.entrySet()) {
          // key = 参数名称(@Param("xxx"))或 参数的索引
          // value = 参数实例
        param.put(entry.getValue(), args[entry.getKey()]);
        
         // add generic param names (param1, param2, ...)
         // 下面就是添加 "paramN" -> 参数实例的映射,我们知道在 mapper 文件中可以使用 #{paramN} 来获取参数列表的值,这就是原因。
        final String genericParamName = GENERIC_NAME_PREFIX + (i + 1);
        // ensure not to overwrite parameter named with @Param
        if (!names.containsValue(genericParamName)) {
          param.put(genericParamName, args[entry.getKey()]);
        }
        i++;
      }
      return param;
    }
  }

那么显然,本次调用返回的 param 如下:

image-20220521101650953

解决方法

通过上面的分析,我们就知道了为什么咱们传给 MybatisParameterHandler#process() 的参数是一个 Map,并且也知道了为什么自动填充失败的根本原因,那么解决方法也就很明确了:

给实体类参数注解为 @Param(“et”),修改后记得 Mapper 文件中占位符中也要改成 #{et.property}

image-20220521102147149

或者方法只有一个实体类参数时就别标注 @Param 注解了,这样返回的就是实体类的实例而不是一个 Map,同样记得 Mapper 文件中占位符直接写属性 #{property} 即可。

image-20220521103258068

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • MybatisX 快速开发插件过程详解

    MybatisX 快速开发插件过程详解

    MybatisX 是一款基于 IDEA 的快速开发插件,方便在使用mybatis以及mybatis-plus开始时简化繁琐的重复操作,提高开发速率。这篇文章主要介绍了MybatisX 快速开发插件,需要的朋友可以参考下
    2021-10-10
  • Java实现鼠标模拟与键盘映射

    Java实现鼠标模拟与键盘映射

    这篇文章主要为大家详细介绍了Java实现鼠标模拟与键盘映射,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-08-08
  • servlet实现用户登录小程序

    servlet实现用户登录小程序

    这篇文章主要为大家详细介绍了servlet实现用户登录的小程序,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05
  • springboot如何获取接口下所有实现类

    springboot如何获取接口下所有实现类

    这篇文章主要介绍了springboot如何获取接口下所有实现类问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-09-09
  • Java使用建造者模式实现办理手机套餐功能详解

    Java使用建造者模式实现办理手机套餐功能详解

    这篇文章主要介绍了Java使用建造者模式实现办理手机套餐功能,较为详细的描述了建造者模式的概念、原理并结合实例形式分析了Java使用建造者模式实现的办理手机套餐功能具体步骤与相关操作注意事项,需要的朋友可以参考下
    2018-05-05
  • java启动如何设置JAR包内存大小

    java启动如何设置JAR包内存大小

    这篇文章主要介绍了java启动如何设置JAR包内存大小问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-02-02
  • SpringBoot定义过滤器、监听器、拦截器的方法

    SpringBoot定义过滤器、监听器、拦截器的方法

    本篇文章主要介绍了SpringBoot定义过滤器、监听器、拦截器的方法,具有一定的参考价值,有兴趣的可以了解一下。
    2017-04-04
  • java获取IP和IP的归属地的方法实践

    java获取IP和IP的归属地的方法实践

    在Java中获取IP地址通常指的是获取本地机器的IP地址或者通过某种方式获取的远程IP地址,本文就来详细的介绍一下,感兴趣的可以了解一下
    2024-05-05
  • Java final static abstract关键字概述

    Java final static abstract关键字概述

    这篇文章主要介绍了Java final static abstract关键字的相关资料,需要的朋友可以参考下
    2016-05-05
  • Springboot解决ajax+自定义headers的跨域请求问题

    Springboot解决ajax+自定义headers的跨域请求问题

    由于浏览器同源策略(同源策略,它是由Netscape提出的一个著名的安全策略,现在所有支持JavaScript 的浏览器都会使用这个策略。接下来通过本文给大家介绍Springboot如何优雅的解决ajax+自定义headers的跨域请求 ,需要的朋友可以参考下
    2019-05-05

最新评论