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

目录
  • association标签多层嵌套问题
    • 排查
    • 从代码上看没有什么问题
    • 正常代码如下
  • 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;

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

(0)

相关推荐

  • MyBatis的嵌套查询解析

    Mybatis表现关联关系比hibernate简单,没有分那么细致one-to-many.many-to-one.one-to-one.而是只有两种association(一).collection(多),表现很简洁.下面通过一个实例,来展示一下Mybatis对于常见的一对多和多对一关系复杂映射是怎样处理的. 以最简单的用户表订单表这个最简单的一对多做示例: 对应的JavaBean: User: public class User { private int id; private String

  • 详解mybatis中association和collection的column传入多个参数问题

    项目中在使用association和collection实现一对一和一对多关系时需要对关系中结果集进行筛选,如果使用懒加载模式,即联合使用select标签时,主sql和关系映射里的sql是分开的,查询参数传递成为问题. mybatis文档: property description column 数据库的列名或者列标签别名.与传递给resultSet.getString(columnName)的参数名称相同.注意: 在处理组合键时,您可以使用column="{prop1=col1,prop2=c

  • 解决mybatis 中collection嵌套collection引发的bug

    我就废话不多说了,大家还是直接看代码吧~ <resultMap id="ParentMap" type="org.example.mybatis.Parent"> <id column="Id" jdbcType="VARCHAR" property="id" /> <result column="Name" jdbcType="VARCHAR&q

  • 解决Mybatis中foreach嵌套使用if标签对象取值的问题

    目录 foreach嵌套使用if标签对象取值问题 大体格式 解决办法 代码如下 Mybatis if 语句嵌套 要求 foreach嵌套使用if标签对象取值问题 最近做项目过程中,涉及到需要在 Mybatis 中 使用 foreach 进行循环读取传入的查询条件,动态拼接SQL语句,接口传入的查询条件格式:{"advanceSearchList":[{"searchType":10,"searchText":"12"}]} ,

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

    目录 association标签多层嵌套问题 排查 从代码上看没有什么问题 正常代码如下 association集合嵌套 这个返回集合有什么用呢 association标签多层嵌套问题 mybatis里查询使用嵌套association标签时,发现内层的association查询的结果一直为null 排查 检查sql执行情况,发现有数据返回,排除 检查property的值是否和pojo中的对应,值一致,排除 检查column的值是否和数据库的相对应,相对应,排除 那么应该是mybatis没有把数

  • Mybatis中Mapper标签总结大全

    一.标签分类 定义SQL语句 insert delete update select 配置关联关系 collection association 配置java对象属性与查询结果集中列名的对应关系 resultMap 控制动态SQL拼接 foreach if choose 格式化输出 where set trim 定义常量 sql 其他 include 二.标签总结 1. 基础SQL标签 1.1 查询select 标签属性 id 唯一的名称,对应dao中mapper的接口名称 paramterTy

  • Mybatis中连接查询和嵌套查询实例代码

    首先在mysql中确立表: #表一:地址国家表 CREATE TABLE address(aid INT AUTO_INCREMENT PRIMARY KEY,aname VARCHAR(20)); INSERT INTO address VALUES(NULL,"魏国"); INSERT INTO address VALUES(NULL,"蜀国"); INSERT INTO address VALUES(NULL,"吴国"); #表二:出场人物

  • 详解mybatis中的if-else的嵌套使用

    目录 案例一:if-else 案例二:if嵌套 MyBatis中if和choose的嵌套 案例一:if-else 在mybatis的使用过程中,难免会存在使用if-else的逻辑,但是实际是没有这种语法的,提供了choose标签来替代这种语法 <select id="selectUserByState" resultType="com.bz.model.entity.User"> SELECT * FROM user WHERE 1=1 <choo

  • MyBatis中association的基本使用方法

    目录 通过association对两表进行联表查询 按照查询嵌套处理 按照结果嵌套处理 总结 通过association对两表进行联表查询 student表属性如下 teacher表属性如下 按照查询嵌套处理 关于需求的SQL稍微有点复杂时,可以打开右侧查询框进行语句的编写执行. 当使用以下时,查询出来存在问题 <select id="getStudentTeacher" resultType="Student" > select s.id,s.name

  • mybatis中resultMap 标签的使用教程

    MyBatis是一个优秀的持久层框架,它对jdbc的操作数据库的过程进行封装,使开发者只需要关注SQL本身,而不需要花费精力去处理例如注册驱动.创建connection.创建statement.手动设置参数.结果集检索等jdbc繁杂的过程代码. MyBatis特点: 1.开源的优秀持久层框架 2.SQL语句与代码分离 3.面向配置的编程 4.良好支持复杂数据映射 5.动态SQL resultMap 标签: 用来描述如何从数据库结果集中来加载对象 (敲黑板!!)主管数据库的字段和实体类属性的匹配,

  • Mybatis中foreach标签带来的空格\换行\回车问题及解决方案

    原因 在自已做的内容中通过获取多个商品Id,以此来获取多个商品详细信息.但数据库返回的商品信息的顺序与原来List中产品的id顺序并不匹配,这就导致了前端页面商品的信息显示混乱,不匹配. 通过网上找到order by排序可以有效解决这个问题,返回与查询时的产品id顺序一致.还有一点,List时有序的 这里先附带上这个sql语句(只能在数据库中成功) select prod_id, prod_name, price, `describe`, prod_date, prod_pic, integra

  • mybatis中bind标签和concat的使用说明

    首先,二种方式都可以用来模糊查询,都能预防 SQL 注入.但是在更换数据库情况下,bind标签通用. <if test=" userName != null and userName !=""> and userName like concat('%' ,#{userName},'%') </if> 使用concat函数连接字符串,在mysql中这个函数支持多个参数,但是在oracle中这个函数只支持2个参数,由于不同数据库之间的语法差异,更换数据库,

  • 基于mybatis中<include>标签的作用说明

    MyBatis中sql标签定义SQL片段,include标签引用,可以复用SQL片段 sql标签中id属性对应include标签中的refid属性.通过include标签将sql片段和原sql片段进行拼接成一个完整的sql语句进行执行. <sql id="sqlid"> res_type_id,res_type </sql> <select id="selectbyId" resultType="com.property.vo

  • mybatis中<if>标签bool值类型为false判断方法

    昨天实现一个功能,根据文章的id或者别名查找文章. 起初采用mybatis的Example进行查询,对参数artName进行判断,如果是纯数字就byId查询,否则就by别名.由于查询文章的同时,需要关联查询文章分类标签,所以选择采用select语句映射的方式查询,但又不想写两个查询方法,就使用了mybatis中动态sql. /** * 查询文章 * @param artName id 或 别名 * @param byId 如果是 true 则按照id查询 * 否则 按照别名查询 * @retur

随机推荐