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的自定义函数确实方便不少,不过与其他的转换工具相比,上手难度确实大,配置也稍显繁琐。
项目代码托管在码云。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对毛票票的支持。如果你想了解更多相关内容请查看下面相关链接