在Mybatis中association标签多层嵌套的问题

 更新时间:2022年03月10日 15:00:47   作者:-半夜微凉。  
这篇文章主要介绍了在Mybatis中association标签多层嵌套的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

association标签多层嵌套问题

mybatis里查询使用嵌套association标签时,发现内层的association查询的结果一直为null

排查

  • 检查sql执行情况,发现有数据返回,排除
  • 检查property的值是否和pojo中的对应,值一致,排除
  • 检查column的值是否和数据库的相对应,相对应,排除

那么应该是mybatis没有把数据映射到位了,经过排查是association中columnPrefix被不对应

<resultMap id="BaseResultMap" type="a.b.c.d.e">
    <id column="id" property="id" />
    <result property="workTime" column="work_time" />
    <result property="model" column="model" />
    <result property="status" column="status" />
    <association property="interfaceUpstream" javaType="interfaceUpstream" columnPrefix="ui_">
      <id column="id" property="id" />
      <result property="interfaceName" column="interface_name" />
      <result property="interfaceType" column="interface_type" />
      <result property="frequency" column="frequency" />
      <result property="address" column="address" />
      <result property="templateOrSql" column="template_or_sql" />
      <result property="status" column="status" />
      <association property="systemInfo" javaType="SystemInfo" columnPrefix="sys_">
        <id column="id" property="id"/>
        <result property="systemName" column="system_name"/>
        <result property="systemNameEN" column="system_name_en"/>
        <result property="belong" column="belong"/>
        <result property="status" column="status"/>
      </association>
      <association property="serverInfo" javaType="ServerInfo" columnPrefix="ser_">
        <id column="id" property="id"/>
        <result property="ftpIp" column="ftp_ip"/>
        <result property="ftpPort" column="ftp_port"/>
        <result property="ftpAccount" column="ftp_account"/>
        <result property="ftpPassword" column="ftp_password"/>
      </association>
    </association>
  </resultMap>
<sql id="base_select">
    SELECT
    ii.Id,
    ii.model,
    ii.status,
    ii.work_time,
    ui.id AS ui_id,
    ui.interface_name AS ui_interface_name,
    ui.interface_type AS ui_interface_type,
    ui.frequency AS ui_frequency,
    ui.address AS ui_address,
    ui.template_or_sql AS ui_template_or_sql,
    ui.status AS ui_status,
    sys.id AS sys_id,
    sys.system_name AS sys_system_name,
    sys.system_name_en AS sys_system_name_en,
    sys.belong AS sys_belong,
    sys.status AS sys_status,
    ser.id AS ser_id,
    ser.ftp_ip AS ser_ftp_ip,
    ser.ftp_port AS ser_ftp_port,
    ser.ftp_account AS ser_ftp_account,
    ser.ftp_password AS ser_ftp_password
  </sql>

从代码上看没有什么问题

原因是association在进行多层嵌套时,mybatis会将外层association的columnPrefix值与内层的进行并合,

如外层columnPrefix值位ui_, 内层为sys_, 那么在SQL中就不能这样 sys.id AS sys_id 了,需要将ui_前缀加上,变成 sys.id AS ui_sys_id ,这样mybatis在匹配的时候才会将数据映射到对应association上

正常代码如下

SELECT
    ii.Id,
    ii.model,
    ii.status,
    ii.work_time,
    ui.id AS ui_id,
    ui.interface_name AS ui_interface_name,
    ui.interface_type AS ui_interface_type,
    ui.frequency AS ui_frequency,
    ui.address AS ui_address,
    ui.template_or_sql AS ui_template_or_sql,
    ui.status AS ui_status,
    sys.id AS ui_sys_id,
    sys.system_name AS ui_sys_system_name,
    sys.system_name_en AS ui_sys_system_name_en,
    sys.belong AS ui_sys_belong,
    sys.status AS ui_sys_status,
    ser.id AS ui_ser_id,
    ser.ftp_ip AS ui_ser_ftp_ip,
    ser.ftp_port AS ui_ser_ftp_port,
    ser.ftp_account AS ui_ser_ftp_account,
    ser.ftp_password AS ui_ser_ftp_password

问题解决! 

association集合嵌套

学了一下mybatis的查询返回值的集合嵌套,先查了查官网:

这个返回集合有什么用呢

举个例子三张表

hr_job_department

hr_job_position

第三张表里在表示部门和职位的时候只用了上面两张表的主键

但是查询的时候,希望表示下面这样的结果

所以返回值是不止一个对象,这样就用到了集合嵌套

<resultMap id="userInfoMap" type="com.advancedc.hrsys.entity.UserInfo">
		<id column="id" property="id" />
		<result column="name" property="name" />
		<result column="gender" property="gender" />
		<result column="id_card" property="idCard" />
		<result column="is_married" property="isMarried" />
		<result column="phone" property="phone" />
		<result column="priority" property="priority" />
		<result column="entry_time" property="entryTime" />
		<result column="full_time" property="fullTime" />
		<result column="created_time" property="createdTime" />
		<result column="edited_time" property="editedTime" />
		<association property="jobDepartment" column="id"
			javaType="com.advancedc.hrsys.entity.JobDepartment">
			<id column="jdid" property="id" />
			<result column="jdname" property="name" />
		</association>
		<association property="jobPosition" column="id"
			javaType="com.advancedc.hrsys.entity.JobPosition">
			<id column="jpid" property="id" />
			<result column="jpname" property="name" />
		</association>
	</resultMap>

只需要知道:

(1)column表示数据库字段

(2)property表示Java里的值

而且我这里的主键都是id所以会出现重名的情况,在SQL语句里,查询时就要赋予别名才能加以区分,返回结果resultMap就如上图所示 

<select id="queryUserInfoBySomeone" resultMap="userInfoMap" resultType="com.advancedc.hrsys.entity.UserInfo">
		SELECT
		ui.id,
		ui.name,
		ui.gender,
		ui.id_card,
		ui.is_married,
		ui.department_id,
		ui.position_id,
		ui.phone,
		ui.priority,
		ui.entry_time,
		ui.full_time,
		ui.created_time,
		ui.edited_time,
		jd.id jdid,
		jd.name jdname,
		jp.id jpid,
		jp.name jpname
		FROM
		hr_user_info ui
		INNER JOIN
		hr_job_department jd
		ON
		ui.department_id=jd.id
		INNER JOIN
		hr_job_position jp
		ON
		ui.position_id=jp.id
		<where>
			<if test="someone.id>0">
			and ui.id = #{someone.id}
			</if>
			<if test="someone.gender!=null">
			and ui.gender = #{someone.gender}
			</if>
			<if test="someone.name!=null">
			and ui.name = #{someone.name}
			</if>
			<if test="someone.idCard!=null">
			and ui.id_card = #{someone.idCard}
			</if>
			<if test="someone.isMarried!=null">
			and ui.is_married = #{someone.isMarried}
			</if>
			<if test="someone.jobDepartment!=null and someone.jobDepartment.id!=null">
			and ui.department_id = #{someone.jobDepartment.id}
			</if>
			<if test="someone.jobPosition!=null and someone.jobPosition.id!=null">
			and ui.position_id = #{someone.jonPosition.id}
			</if>
			<if test="someone.phone!=null">
			and ui.phone = #{someone.phone}
			</if>
			<if test="someone.entryTime!=null">
			and ui.entry_time = #{someone.entryTime}
			</if>
			<if test="someone.fullTime!=null">
			and ui.full_time = #{someone.fullTime}
			</if>
		</where>
	</select>

上图用了INNER JOIN来查询看上去挺简洁的,有一种不简洁的写法如下,虽然也能得到结果,但是不知道性能对比如何

SELECT
		ui.id,
		ui.name,
		ui.gender,
		ui.id_card,
		ui.is_married,
		ui.department_id,
		ui.position_id,
		ui.phone,
		ui.priority,
		ui.entry_time,
		ui.full_time,
		ui.created_time,
		ui.edited_time,
		(select id jdid from hr_job_department jd where jd.id=ui.department_id) jdid,
		(select name jdname from hr_job_department jd where jd.id=ui.department_id) jdname,
		(select id jpid from hr_job_position jp where jp.id=ui.position_id) jpid,
		(select name jpname from hr_job_position jp where jp.id=ui.position_id) jpname
		FROM
		hr_user_info ui;

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

相关文章

  • Java实战之城市多音字处理

    Java实战之城市多音字处理

    这篇文章主要介绍了Java实战之城市多音字处理,文中有非常详细的代码示例,对正在学习java的小伙伴们有非常好的帮助,需要的朋友可以参考下
    2021-04-04
  • 解决nacos的yml配置文件解析@开头的值启动报错问题

    解决nacos的yml配置文件解析@开头的值启动报错问题

    这篇文章主要介绍了解决nacos的yml配置文件解析@开头的值启动报错问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-07-07
  • JAVA中的函数式接口Function和BiFunction详解

    JAVA中的函数式接口Function和BiFunction详解

    这篇文章主要介绍了JAVA中的函数式接口Function和BiFunction详解,JDK的函数式接口都加上了@FunctionalInterface注解进行标识,但是无论是否加上该注解只要接口中只有一个抽象方法,都是函数式接口,需要的朋友可以参考下
    2024-01-01
  • sprintboot使用spring-security包,缓存内存与redis共存方式

    sprintboot使用spring-security包,缓存内存与redis共存方式

    这篇文章主要介绍了sprintboot使用spring-security包,缓存内存与redis共存方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • Java实战之图书管理系统的实现

    Java实战之图书管理系统的实现

    这篇文章主要介绍了如何利用Java语言编写一个图书管理系统,文中采用的技术有Springboot、SpringMVC、MyBatis、ThymeLeaf 等,需要的可以参考一下
    2022-03-03
  • Java中如何将String转JSONObject

    Java中如何将String转JSONObject

    这篇文章主要介绍了Java中如何将String转JSONObject,String类型转JSONObject,下面有两种方式可以进行转换,本文结合实例代码给大家介绍的非常详细,需要的朋友可以参考下
    2023-05-05
  • Springboot Filter中注入bean无效为null问题

    Springboot Filter中注入bean无效为null问题

    这篇文章主要介绍了Springboot Filter中注入bean无效为null问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-05-05
  • SpringBoot实现联表查询的代码详解

    SpringBoot实现联表查询的代码详解

    这篇文章主要介绍了SpringBoot中如何实现联表查询,文中通过代码示例和图文结合的方式讲解的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下
    2024-05-05
  • Java8接口中引入default关键字的本质原因详析

    Java8接口中引入default关键字的本质原因详析

    Default方法是在java8中引入的关键字,也可称为Virtual extension methods—虚拟扩展方法,这篇文章主要给大家介绍了关于Java8接口中引入default关键字的本质原因,需要的朋友可以参考下
    2022-01-01
  • SpringBoot事件发布和监听详解

    SpringBoot事件发布和监听详解

    今天去官网查看spring boot资料时,在特性中看见了系统的事件及监听章节,所以下面这篇文章主要给大家介绍了关于SpringBoot事件发布和监听的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2021-11-11

最新评论