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这个词的意义。