我的编程空间,编程开发者的网络收藏夹
学习永远不晚

mybatis主从表关联查询,返回对象带有集合属性解析

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

mybatis主从表关联查询,返回对象带有集合属性解析

主从表关联查询,返回对象带有集合属性

昨天有同事让我帮着看一个问题,mybatis主从表联合查询,返回的对象封装集合属性。我先将出现的问题记录一下,然后再讲处理方法也简单说明一下:

VersionResult为接收返回数据对象

get\set方法我这里就省略了。

public class VersionResult extends BaseResult implements Serializable{
	private Integer id;
	private String code;
	@JsonFormat(pattern = "yyyy-MM-dd HH:mm", timezone = "GMT+8")
	private Date createTimes;
	//记录内容表的集合对象
	private List<UpdateRecordEntity> UpdateRecordEntityList;	
	}

UpdateRecordEntity为从表数据

同样get\set方法我这里就省略了。

@Table(name = "z_update_record")
public class UpdateRecordEntity extends BaseEntity {
    @Id
    private Integer id;
    @Column(name = "version_id")
    private Integer versionId;
    @Column(name = "module_name")
    private String moduleName;
    @Column(name = "update_content")
    private String updateContent;
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm", timezone = "GMT+8")
    @Column(name = "create_time")
    private Date createTime;
    @Column(name = "is_delete")
    private Integer isDelete;
    }

mapper.xml写法,这个是关键

 <!--跟新记录表封装的对象-->
 <resultMap id="BaseResultMap" type="com.wangtiansoft.wisdomedu.persistence.result.server.VersionResult">
     <id column="id" property="id" jdbcType="INTEGER"/>
     <result column="code" property="code" />
     <result column="create_time" property="createTimes" />
     <collection property="UpdateRecordEntityList" ofType="com.wangtiansoft.wisdomedu.persistence.entity.UpdateRecordEntity">
         <id property="id" column="id"/>
         <result property="versionId" column="version_id"/>
         <result property="moduleName" column="module_name"/>
         <result property="updateContent"  column="update_content"/>
         <result property="createTime" column="create_time"/>
         <result property="isDelete" column="is_delete"/>
         <result property="tenantId" column="tenant_id"/>
     </collection>
 </resultMap>

sql查询语句

<select id="selectVersionList" parameterType="map" resultMap="BaseResultMap">
     SELECT
         z.`code`,
         z.create_time createTimes,
         zur.module_name moduleName,
         zur.update_content updateContent,
         zur.create_time createTime
     FROM
      	 z_version z
     LEFT JOIN z_update_record zur ON z.id = zur.version_id
     WHERE
         z.tenant_id = #{tenantId}
         AND z.is_delete = 0
         AND z.is_disabled = 0
         AND zur.tenant_id = #{tenantId}
         AND zur.is_delete = 0
         AND YEAR(z.create_time)=YEAR(#{date})
         ORDER by z.create_time desc
 </select>

执行sql返回的数据

页面调取接口

下面我将接口数据粘贴下来:

{
	"code": "0",
	"msg": "",
	"data": [{
		"id": null,
		"code": "1419",
		"createTimes": null,
		"updateRecordEntityList": []
	}, {
		"id": null,
		"code": "开发修改1111",
		"createTimes": null,
		"updateRecordEntityList": []
	}, {
		"id": null,
		"code": "开发修改1111",
		"createTimes": null,
		"updateRecordEntityList": []
	}, {
		"id": null,
		"code": "开发修改1111",
		"createTimes": null,
		"updateRecordEntityList": []
	}, {
		"id": null,
		"code": "开发修改1111",
		"createTimes": null,
		"updateRecordEntityList": []
	}]
}

观察code、createTimes、updateRecordEntityList三个属性,会发现只有code字段有值其余的全部为null。分析这个是为啥呢?找点资料粘贴如下:

发现是sql数据和VersionResult的mapper.xml中映射关系有点问题,没有对应起来。resultMap中必须将别名和上面resultMap对的上就行,很明显sql返回数据的列明没有和resultMap一一对应起来,因此有了以下对xml文件的修改:

   <resultMap id="BaseResultMap" type="com.wangtiansoft.wisdomedu.persistence.result.server.VersionResult">
        <id column="id" property="id" jdbcType="INTEGER"/>
        <result column="code" property="code" />
        <result column="createTimes" property="createTimes" />
        <collection property="UpdateRecordEntityList" ofType="com.wangtiansoft.wisdomedu.persistence.entity.UpdateRecordEntity">
            <id property="id" column="id"/>
            <result property="moduleName" column="moduleName"/>
            <result property="updateContent"  column="updateContent"/>
            <result property="createTime" column="createTime"/>
        </collection>
    </resultMap>

数据显示正常:

{
    "code": "0",
    "msg": "",
    "data": [{
        "code": "1419",
        "createTimes": "2019-09-02 00:00",
        "updateRecordEntityList": [{
            "moduleName": "安达市大所",
            "updateContent": "1321321",
            "createTime": "2019-09-02 10:17"
        }]
    }, {
        "code": "开发修改1111",
        "createTimes": "2019-05-07 00:00",
        "updateRecordEntityList": [{
            "moduleName": "平台111111",
            "updateContent": "平台版本第一次更新1",
            "createTime": "2019-08-15 15:07"
        }]
    }, {
        "code": "开发修改1111",
        "createTimes": "2019-05-07 00:00",
        "updateRecordEntityList": [{
            "moduleName": "111",
            "updateContent": "111",
            "createTime": "2019-08-16 11:16"
        }]
    }, {
        "code": "开发修改1111",
        "createTimes": "2019-05-07 00:00",
        "updateRecordEntityList": [{
            "moduleName": "515",
            "updateContent": "5155",
            "createTime": "2019-08-21 17:29"
        }]
    }, {
        "code": "开发修改1111",
        "createTimes": "2019-05-07 00:00",
        "updateRecordEntityList": [{
            "moduleName": "2222",
            "updateContent": "第二次更新",
            "createTime": "2019-08-22 14:23"
        }]
    }]
}

mybatis关联查询(对象嵌套对象)

Mybatis 查询对象中嵌套其他对象的解决方法有两种,

一种是用关联另一个resultMap的形式

如下:

<association property="office"  javaType="Office" resultMap="officeMap"/>
<mapper namespace="com.dixn.oa.modules.sys.dao.RoleDao">
    
   <resultMap type="Office" id="officeMap">
        <id property="id" column="id" />
        <result property="name" column="office.name" />
        <result property="code" column="office.code" />
    </resultMap>
    
    <resultMap id="roleResult" type="Role">
        <id property="id" column="id" />
        <result property="name" column="name" />
        <result property="enname" column="enname" />
        <result property="roleType" column="roleType" />
        <result property="dataScope" column="dataScope" />
        <result property="remarks" column="remarks" />
        <result property="useable" column="useable" />
        <association property="office"  javaType="Office" resultMap="officeMap"/>
        <collection property="menuList" ofType="Menu">
            <id property="id" column="menuList.id" />
        </collection>
        <collection property="officeList" ofType="Office">
            <id property="id" column="officeList.id" />
        </collection>
    </resultMap>
    <sql id="roleColumns">
        a.id,
        a.office_id AS "office.id",
        a.name,
        a.enname,
        a.role_type AS roleType,
    a.data_scope AS dataScope,
    a.remarks,
    a.create_by AS "createBy.id",
    a.create_date,
    a.update_by AS "updateBy.id",
    a.update_date,
    a.del_flag,
        o.name AS "office.name",
        o.code AS "office.code",
        a.useable AS useable,
        a.is_sys AS sysData
    </sql>
<select id="get" resultMap="roleResult">
    SELECT
    <include refid="roleColumns"/>
    rm.menu_id AS "menuList.id",
    ro.office_id AS "officeList.id"
    FROM sys_role a
    JOIN sys_office o ON o.id = a.office_id
    LEFT JOIN sys_role_menu rm ON rm.role_id = a.id
    LEFT JOIN sys_role_office ro ON ro.role_id = a.id
    WHERE a.id = #{id}
</select>

一种联合查询 (一对一)的实现

但是这种方式有“N+1”的问题,不建议使用

 <resultMap id="roleResult" type="Role">
        <id property="id" column="id" />
        <result property="name" column="name" />
        <result property="enname" column="enname" />
        <result property="roleType" column="roleType" />
        <result property="dataScope" column="dataScope" />
        <result property="remarks" column="remarks" />
        <result property="useable" column="useable" />
        <association property="office"  javaType="Office"     column="id" select="getOfficeById"/>
        <collection property="menuList" ofType="Menu">
            <id property="id" column="menuList.id" />
        </collection>
        <collection property="officeList" ofType="Office">
            <id property="id" column="officeList.id" />
        </collection>
    </resultMap>
    <select id="getOfficeById"  resultType="Office">
        select o.name AS "office.name",o.code AS "office.code" from sys_office o where o.id = #{id}
    </select> 

以上就是两种对象内嵌套对象查询的实现。仅为个人经验,希望能给大家一个参考,也希望大家多多支持编程网。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

mybatis主从表关联查询,返回对象带有集合属性解析

下载Word文档到电脑,方便收藏和打印~

下载Word文档

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录