在这篇文章中,我们将进入Mybatis注解的世界,了解实现基于注解的结果映射的基本方法,其中的很多观点或内容都能在一定程度上让我们的开发之旅更加轻松方便,这是一个菜鸟提升技术能力,老鸟巩固基础知识的好机会。准备好开启今天的神奇之旅了吗?
大家好,我是奇迹老李,一个专注于分享开发经验和基础教程的博主。这里是我的其中一个技术分享平台,欢迎广大朋友们点赞评论提出意见,重要的是点击关注喔 🙆。今天要和大家分享的内容是基于注解的结果映射。做好准备,Let’s go🚎🚀
在resultMap标签中常用的元素都有对应等价形式的java注解,这在Mybatis官方文档中有明确的说明
注解 | xml等价形式 | 描述 |
---|---|---|
@Results | <resultMap> | 映射一系列的字段和属性 |
@Result | <result>,<id> | 映射一个字段和属性 |
@ConstructorArgs | <constructor> | 将结果映射到对象的构造方法 |
@Arg | <arg>,<idArg> | 构造方法的一个参数 |
@One | <association> | 一对一关联 |
@Many | <collection> | 一对多关联 |
@Insert | <insert> | 指定新增的sql |
@Delete | <delete> | 指定删除的sql |
@Update | <update> | 指定更新的sql |
@Select | <select> | 指定查询的sql |
@InsertProvider | <insert> | 指定新增的sql provider |
@DeleteProvider | <delete> | 指定删除的sql provider |
@UpdaateProvider | <update> | 指定更新的sql provider |
@SelectProvider | <select> | 指定查询的sql provider |
@ResultMap | 没有 | 指定使用xml文件中的resultMap |
@ResultType | 没有 | 当方法返回值为空时,指定返回值类型 |
@MapKey | 没有 | 将方法的List返回值,转化为一个Map,其中Map的key就是@MapKey(value="") 中的内容 |
注册一个新的mapper
<configuration>
<mappers>
<mapper resource="mapper/ApplicationMapper.xml"/>
<mapper resource="mapper/SimpleQueryMapper.xml"/>
<mapper class="top.sunyog.mybatis.mapper.AppAnnoMapper"/>
</mappers>
</configuration>
一个简单的查询sql写法是这样的
public interface AppAnnoMapper {
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
Map<String,Object> selectAppEntityById(Long id);
}
测试类代码:
public class AppAnnoService extends MybatisService<AppAnnoMapper> {
private AppAnnoMapper mapper;
@Override
public void doService() {
this.mapper = this.getMapper(AppAnnoMapper.class);
this.testSelectResult();
}
private void testSelectResult() {
Map<String, Object> res = this.mapper.selectAppEntityById(1L);
res.entrySet().forEach(e-> System.out.println(e.getKey()+":=> "+e.getValue()));
}
}
打印输出查询结果如下:
app_name:=> 测试应用1
auth_type:=> 1
creator:=> admin
app_status:=> 8
id:=> 1
create_date:=> 2023-10-31
app_code:=> ceshi
可以通过 @Result注解转换查询结果的属性名称,代码如下:
@Result(column = "app_name",property = "APPLICATION_NAME")
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
Map<String,Object> selectAppEntityById(Long id);
效果如下,app_name属性的名称已经修改
auth_type:=> 1
creator:=> admin
app_status:=> 8
id:=> 1
create_date:=> 2023-10-31
APPLICATION_NAME:=> 测试应用1
app_code:=> ceshi
如果多个属性的名称都需要重新映射,则使用 @Results
@Results({
@Result(id = true,column = "id",property = "APP_ID"),
@Result(column = "app_name",property = "APPLICATION_NAME"),
@Result(column = "app_code",property = "APPLICATION_CODE"),
@Result(column = "auth_type",property = "AUTHENTICATION_TYPE"),
@Result(column = "creator",property = "CREAT_BY"),
@Result(column = "app_status",property = "STATUS"),
@Result(column = "create_date",property = "BIRTHDAY")
})
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
Map<String,Object> selectAppEntityById(Long id);
效果如下:
APP_ID:=> 1
STATUS:=> 8
AUTHENTICATION_TYPE:=> 1
CREAT_BY:=> admin
APPLICATION_NAME:=> 测试应用1
BIRTHDAY:=> 2023-10-31
APPLICATION_CODE:=> ceshi
和 <resultMap>
标签一样,@Result
注解也支持自定义类型转换器,写法如下
//mapper接口代码
@Result(column = "create_date",property = "BIRTHDAY",typeHandler = LocalDateTypeHandler.class)
//测试方法代码
private void testSelectResult() {
Map<String, Object> res = this.mapper.selectAppEntityById(1L);
System.out.println(res.get("BIRTHDAY").getClass());
}
修改后的打印结果:class java.time.LocalDate
修改前的打印结果:class java.sql.Date
这两个注解的作用也是映射结果属性,区别在于 @ConstructorArgs用于通过构造函数映射,而 @Resutls用于setter方法映射。构造函数映射需要注意定义的参数的顺序要和构造函数入参的顺序一致。
public class ApplicationEntity implements Serializable {
private Long id;
private String name;
private String code;
private String auth;
private LocalDate birthday;
private String creator;
private String status;
public ApplicationEntity( Long id, String name, String code, String auth, LocalDate birthday, String creator
, String status) {
this.id = id;
this.name = name;
this.code = code;
this.auth = auth;
this.birthday = birthday;
this.creator = creator;
this.status = status;
}
@Override
public String toString() {
final StringBuffer sb = new StringBuffer("ApplicationEntity{");
sb.append("id=").append(id);
sb.append(", name='").append(name).append('\'');
sb.append(", code='").append(code).append('\'');
sb.append(", auth='").append(auth).append('\'');
sb.append(", birthday=").append(birthday);
sb.append(", creator='").append(creator).append('\'');
sb.append(", status='").append(status).append('\'');
sb.append('}');
return sb.toString();
}
}
@ConstructorArgs({
//这里的顺序需要和实体类的构造函数入参顺序一致
@Arg(id = true, column = "id", javaType = Long.class),
@Arg(column = "app_name", javaType = String.class),
@Arg(column = "app_code", javaType = String.class),
@Arg(column = "auth_type", javaType = String.class),
@Arg(column = "create_date", javaType = LocalDate.class),
@Arg(column = "creator", javaType = String.class),
@Arg(column = "app_status", javaType = String.class)
})
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
ApplicationEntity selectAppArgsById(Long id);
private void testSelectConstructor(){
ApplicationEntity entity = this.mapper.selectAppArgsById(1L);
System.out.println(entity);
}
打印结果如下:
ApplicationEntity{id=1, name='测试应用1', code='ceshi', auth='1', birthday=2023-10-31, creator='admin', status='8'}
这时,如果调换@Arg注解的先后位置,如下
@ConstructorArgs({
@Arg(id = true, column = "id", javaType = Long.class),
@Arg(column = "app_code", javaType = String.class),
@Arg(column = "app_name", javaType = String.class),
@Arg(column = "auth_type", javaType = String.class),
@Arg(column = "create_date", javaType = LocalDate.class),
@Arg(column = "creator", javaType = String.class),
@Arg(column = "app_status", javaType = String.class)
})
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
ApplicationEntity selectAppArgsById(Long id);
就会产生这样的结果,name和code的映射结果是错误的
ApplicationEntity{id=1, name='ceshi', code='测试应用1', auth='1', birthday=2023-10-31, creator='admin', status='8'}
注意:@Arg
注解有一个name属性,这个属性在 @ConstructorArgs
注解中不能出现,如果出现,构造函数会映射失败导致报错。想要解决这个问题需要在实体类的构造函数中通过 @Param
注解指定每个入参的名称和@Arg
注解中的name属性一一对应,如下所示:
实体类:
public class ApplicationEntity implements Serializable {
//省略属性
public ApplicationEntity(@Param("id") Long id,@Param("name") String name,@Param("code") String code
,@Param("auth") String auth,@Param("birthday") LocalDate birthday,@Param("creator") String creator
,@Param("status") String status) {
...
}
//省略其他方法
}
mapper接口
@ConstructorArgs({
@Arg(id = true, column = "id", javaType = Long.class, name = "id"),
@Arg(column = "app_code", javaType = String.class, name = "code"),
@Arg(column = "app_name", javaType = String.class, name = "name"),
@Arg(column = "auth_type", javaType = String.class, name = "auth"),
@Arg(column = "create_date", javaType = LocalDate.class, name = "birthday"),
@Arg(column = "creator", javaType = String.class, name = "creator"),
@Arg(column = "app_status", javaType = String.class, name = "status")
})
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
ApplicationEntity selectAppArgsById(Long id);
这样就改变了 ResultMap的映射方式(从按顺序改为按名称),即便修改了@Arg
的定义顺序,也能正确的映射到对应的属性上
改造AppAnnoMapper#selectAppEntityById
方法使其能够达成 <association>
标签的效果
public interface AppAnnoMapper {
@Results(id = "APP_MAP", value = {
@Result(id = true, column = "id", property = "APP_ID"),
@Result(column = "app_name", property = "APPLICATION_NAME"),
@Result(column = "app_code", property = "APPLICATION_CODE"),
@Result(column = "auth_type", property = "AUTHENTICATION_TYPE"),
@Result(column = "creator", property = "CREAT_BY"),
@Result(column = "app_status", property = "STATUS"),
@Result(column = "create_date", property = "BIRTHDAY", typeHandler = LocalDateTypeHandler.class),
//添加一个一对一关系,映射到selectAppStatus方法
@Result(column = "app_status",property = "STATUS_DICT",one = @One(select = "selectAppStatus"))
})
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
Map<String, Object> selectAppEntityById(Long id);
//添加状态字典查询
@Select("select dict_name, dict_code, dict_type, dict_sort from dict_test where dict_type='app_status' and dict_code=#{code}")
Map<String,Object> selectAppStatus(@Param("code")String code);
}
原方法的打印结果如下,其中STATUS_DICT属性为 @One关联出的结果
APP_ID:=> 2
STATUS:=> 0
STATUS_DICT:=> [{dict_type=app_status, dict_name=临时应用, dict_sort=0, dict_code=0}]
AUTHENTICATION_TYPE:=> 2
CREAT_BY:=> admin2
APPLICATION_NAME:=> 公共应用1
BIRTHDAY:=> 2023-10-31
APPLICATION_CODE:=> common
<collection>
的效果改动如下
public interface AppAnnoMapper {
@Results(id = "APP_MAP", value = {
@Result(id = true, column = "id", property = "APP_ID"),
@Result(column = "app_name", property = "APPLICATION_NAME"),
@Result(column = "app_code", property = "APPLICATION_CODE"),
@Result(column = "auth_type", property = "AUTHENTICATION_TYPE"),
@Result(column = "creator", property = "CREAT_BY"),
@Result(column = "app_status", property = "STATUS"),
@Result(column = "create_date", property = "BIRTHDAY", typeHandler = LocalDateTypeHandler.class),
@Result(column = "app_status",property = "STATUS_DICT",one = @One(select = "selectAppStatus")),
//添加一个一对多关系,映射到selectAppServices方法
@Result(column = "id",property = "SERVICE_LIST",many = @Many(select = "selectAppServices"))
})
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
Map<String, Object> selectAppEntityById(Long id);
@Select("select id, service_name, service_code, service_path, app_id from service_test where app_id=#{appId}")
List<ServiceTestEntity> selectAppServices(Long appId);
@Select("select dict_name, dict_code, dict_type, dict_sort from dict_test where dict_type='app_status' and dict_code=#{code}")
Map<String,Object> selectAppStatus(@Param("code")String code);
}
可以在映射接口中使用xml文件中定义的resultMap,使用 @ResultMap注解实现
配置文件中新增mapper映射文件配置
<configuration>
...
<mappers>
<mapper resource="mapper/ApplicationMapper.xml"/>
<mapper resource="mapper/SimpleQueryMapper.xml"/>
<mapper class="top.sunyog.mybatis.mapper.AppAnnoMapper"/>
<mapper resource="mapper/AppAnnoMapper.xml"/>
</mappers>
</configuration>
新增xml映射文件
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"https://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="top.sunyog.mybatis.mapper.AppAnnoMapper">
<resultMap id="applicationEntityResultMap" type="top.sunyog.common.entity.ApplicationEntity">
<constructor>
<idArg column="id" javaType="Long"/>
<arg column="app_name" javaType="String"/>
<arg column="app_code" javaType="String"/>
<arg column="auth_type" javaType="String"/>
<arg column="create_date" javaType="java.time.LocalDate"/>
<arg column="creator" javaType="String"/>
<arg column="app_status" javaType="String"/>
</constructor>
</resultMap>
</mapper>
mapper接口新增测试方法
@ResultMap(value = "applicationEntityResultMap")
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
ApplicationEntity selectAppEntityMap(Long id);
测试代码
public class AppAnnoService extends MybatisService<AppAnnoMapper> {
private AppAnnoMapper mapper;
...
private void testAnnoResultMap(){
ApplicationEntity entity = this.mapper.selectAppEntityMap(2L);
System.out.println(entity);
}
}
打印结果如下:
ApplicationEntity{id=2, name='公共应用1', code='common', auth='2', birthday=2023-10-31, creator='admin2', status='0'}
@ResultType注解和 xml文件中的<select id="" resultType="">
属性的作用是完全不同的,@ResultType注解的作用是在mapper类中的方法中传入了 ResultHandler<?>
参数时,说明结果查询结果的类型。
这时,如果不使用@ResultType说明结果类型,ResultHandler也就无法使用了。它的用法如下:
@ResultType(ApplicationEntity.class)
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id=#{id}")
void selectAppEntityType(Long id, ResultHandler<ApplicationEntity> handler);
测试代码:
private void testAnnoType(){
this.mapper.selectAppEntityType(2L, new ResultHandler<ApplicationEntity>() {
@Override
public void handleResult(ResultContext<? extends ApplicationEntity> resultContext) {
ApplicationEntity entity = resultContext.getResultObject();
System.out.println(entity);
}
});
}
打印结果如下:
ApplicationEntity{id=2, name='公共应用1', code='common', auth='2', birthday=2023-10-31, creator='admin2', status='0'}
基于注解可以实现一个xml映射文件中不支持的功能,即将列表返回值转换成一个Map,其中Map的key可以通过 @MapKey
注解控制,实现方法如下:
public interface AppAnnoMapper {
@MapKey(value = "id")
@Select("select id, app_name, app_code, auth_type, create_date, creator, app_status from app_test where id >= #{id}")
Map<Long,ApplicationEntity> selectAppListMap(Long id);
}
测试代码
private void testMapKey(){
Map<Long, ApplicationEntity> map = this.mapper.selectAppListMap(2L);
map.entrySet().forEach(o->{
System.out.println(o.getKey()+":=> "+o.getValue());
});
}
打印结果如下
2:=> ApplicationEntity{id=2, name='公共应用1', code='common', auth='2', birthday=2023-10-31, creator='admin2', status='0'}
5:=> ApplicationEntity{id=5, name='名称1', code='code-1', auth='2', birthday=2023-11-03, creator='admin3', status='0'}
6:=> ApplicationEntity{id=6, name='name1', code='code-2', auth='2', birthday=2023-11-04, creator='admin3', status='0'}
7:=> ApplicationEntity{id=7, name='jack liu', code='code-3', auth='2', birthday=2023-11-04, creator='admin3', status='0'}
在这片文章中,我们介绍了Mybatis基于注解方式的结果映射,Mybatis的结果映射的注解主要有以下几个
其中,@MapKey
还能将查询结果重新构建为一个HashMap,这在xml映射文件中是不支持的操作。
📩 联系方式
邮箱:qijilaoli@foxmail.com?版权声明
本文为原创文章,版权归作者所有。未经许可,禁止转载。更多内容请访问奇迹老李的博客首页