MapStruct处理Java中实体与模型间不匹配属性转换的方法
摘要:前面介绍了MapStrut简单用法,MapStrut的最重要的特点就是处理Java中实体与模型间不匹配属性的转换。
实体模型
有一个User对象:
publicclassUser{
privateIntegerid;
privateStringname;
privatedoubleaccount;
privatebooleanmarried;
//setters,getters,toString()
}
有一个Employee对象:
publicclassEmployee{
privateintid;
privateStringename;
privateStringposition;
privateStringmarried;
//setters,getters,toString()
}
业务场景
- 需要User与Employee对象之间转换。
- User的name属性对应Employee的ename属性,其取值相同,类型相同,名称不同
- User的married属性(取值true和false)对应Employee的married属性(取值Y和N),其取值不同,类型不同,名称相同。
分析与实现
最愚蠢的方式是自己写一堆的setter方法与getter方法,大量get/set代码堆积,增加了代码长度和阅读代码的难度。利用工具BeanUtils是可以处理第一个需求的,但第三种需求就无能为力了。这时MapStrut就派上用场了,最简单的配置可以像下面这样:
@Mapper
publicinterfaceUserMapper{
UserMapperINSTANCE=Mappers.getMapper(UserMapper.class);
EmployeeuserToEmployee(Useruser);
UseremployeeToUser(Employeeemployee);
}
对于第二个需求,可以通过下面方式实现,注解@Mapping可以指定需要把哪个字段source转换为哪个字段target。
@Mapper
publicinterfaceUserMapper{
UserMapperINSTANCE=Mappers.getMapper(UserMapper.class);
@Mappings({
@Mapping(source="name",target="ename")
})
EmployeeuserToEmployee(Useruser);
@Mappings({
@Mapping(source="ename",target="name")
})
UseremployeeToUser(Employeeemployee);
}
第三个需求有点变态,但是真实发生在我们的项目中,实现起来确实繁琐一些:
首先,自定义转化逻辑,布尔值到字符串,布尔的true对应字符串的Y,布尔的false对应字符串的N:
publicclassUserTransform{
publicStringbooleanToString(booleanvalue){
if(value){
return"Y";
}
return"N";
}
publicbooleanstrToBoolean(Stringstr){
if("Y".equals(str)){
returntrue;
}
returnfalse;
}
}
使用很简单,在接口的注解Mapper添加uses参数,值就是需要刚才的转换逻辑类。
@Mapper(uses=UserTransform.class)
publicinterfaceUserMapper{...}
结果与分析
用JunitTest写两个测试方法,分别测试User对象转换Employee,Employee对象转换User。
publicclassMidTest{
@Test
publicvoidmidTest(){
Useruser=newUser();
user.setId(125);
user.setName("Lee");
user.setMarried(true);
Employeee=UserMapper.INSTANCE.userToEmployee(user);
System.out.println(e);
}
@Test
publicvoidmidTest2(){
Employeee=newEmployee();
e.setId(222);
e.setEname("Chao");
e.setMarried("N");
Useru=UserMapper.INSTANCE.employeeToUser(e);
System.out.println(u);
}
}
结果如下:
User[id=222,name=Chao,account=0.0,married=false]
Employee[id=125,ename=Lee,position=null,married=Y]
转换结果符合预期,转化期间不存在的属性,有了默认值(account和position),包装类也能识别(int和Integer),从自动生成的UserMapperImpl.java中,可以看到,
employee.setMarried(userTransform.booleanToString(user.isMarried()));,用到了刚才自定义的转换逻辑。第三种需求是很少的,但是遇到了也是很难解决的,MapStruct的自定义函数确实方便不少,不过与其他的转换工具相比,上手难度确实大,配置也稍显繁琐。
项目代码托管在码云。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对毛票票的支持。如果你想了解更多相关内容请查看下面相关链接