Mybatis的< resultMap>中 < id> 标签的作用(用于collection)

resultMap的作用

<resultMap>标签用于封装sql的查询结果,可以包装成一个简单POJO对象,也可以包装成我们自定义的对象,只要我们使用<result>子标签指定好查询结果的列和对象的属性之间的对应关系就好了。

官方原因

说是用于提高性能,但是在一些情况下,没有Id的话结果会出错。
官方doc传送门(中文):https://mybatis.org/mybatis-3/zh/sqlmap-xml.html
在这里插入图片描述

问题发现:

给出这样一个场景,sql查询每个国家下的用户数,第一列是用户数,后面三列是一个国家对象所含的信息。
在这里插入图片描述
现在我要把结果集封装成如下自定义的对象集合。
其中的Country是一个简单类,有三个属性,和上图查询结果的后三列同名。

public class MyCountResult {
   private Long count;
   private Country country;
    }

在mapper文件中自定义resultMap(红色是因为省略了很长的限定名)
在这里插入图片描述
这样确实能把上面sql查询图的结果封装成 MyCountResult
列count 对应 类的count属性
剩下的列 对应 类的coutry嵌套类 (这里的列也复用了别的resultMap)

但是
返回的MyCountResult 对象却只有3个,而不是上面查询结果图中的十几个。
调试发现,count为0的对象只有1个!
这里是为什么呢?

原因

<collection>标签,用于包装成集合,默认把id相同的封装在同一个集合内

在这里,由于没有指定 < id > 标签,导致mybatis把我的count列当成了id,而很不幸,count列有很多重复的0,所以把所有count为0的行封装成了一个集合,注入到MyCountResult类中,但是这个类中的country属性不是一个数组,所以mybatis只能把刚才的集合中的其中一个注入成country属性。

结语与反思:

既然官方要我们加入 id 标签,无论是性能还是正确性,我们都应该加入此标签。
而且作为id的列中的值必须是唯一的,也就是可以唯一标识这一行,例如刚才我的count列就有很多相同的0,也就失去了id这个词的意义。

Mybatis的<resultMap>标签用于将查询结果集的列与Java对象属性之间进行映射。使用<resultMap>标签可以灵活地对查询结果进行映射,可以将一个查询结果映射到多个Java对象,也可以将多个查询结果映射到一个Java对象。 <resultMap>标签的基本语法如下: ```xml <resultMap id="resultMap" type="javaType"> <!-- Result property --> <result property="propertyName" column="columnName" /> <!-- Other Result property element --> </resultMap> ``` 其id属性为<resultMap>标签的唯一标识符,type属性指定映射的Java类型。 <result>标签用于指定一个Java对象属性和查询结果集的一列之间的映射关系。其,property属性指定Java对象属性的名称,column属性指定查询结果集的列名。 <resultMap>标签还支持一些高级属性,例如:继承其他<resultMap>标签、自动映射、高级映射等。 继承其他<resultMap>标签可以使得<resultMap>之间形成继承关系,从而减少重复的映射定义。 自动映射可以根据Java对象属性的名称和查询结果集的列名之间的相似度自动生成映射关系。 高级映射可以将查询结果集的一列映射到Java对象的一个属性,并且该属性的值是由多个列合并而成的,可以使用<association>和<collection>标签实现。 总的来说,<resultMap>标签Mybatis非常重要的一个标签,掌握其基本语法和高级用法对于开发高效的Mybatis应用非常有帮助。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值