在Java的世界中,经常会涉及到需要在2个对象中进行转换,比如说:
- 调用SOAP Web服务,需要把自己的Domain对象转换为Soap服务的Jaxb对象请求,
- 在分层级SOA架构中,2个层级之间Domain对象的转换,
- 在分布式系统中,每个模块使用自己的领域对象,防止自己的业务被其他模块的domain对象影响
通常情况下,程序员们会自己弄一个Transformer层,来做对象之间的转换。估计很多同志在自己的项目代码中都可以看到Transformer的身影。但是,复杂的Java对象之间的转换并不是一个非常容易的问题,因为你需要不停的检测某个属性值是否为null,你需要做属性类型的转换,Date对应到String, String对应到Int等等。这就导致了Transformer的臃肿以及对程序员生产力的浪费。Dozer就是对懒(you)惰(xiu)程序员在这个问题上的通用解决方案。
Dozer是一个开源的,通用的,健壮的,灵活的,易于重用的,可配置的Java对象Mapping框架 。这个定义中的每一个形容词对程序员都非常亲切. (作者注: Dozer虽好,请不要贪恋, 因为任何对象之间的转换都会带来维护和性能的开销,能不做对象转换还是不要做了,跑不掉的时候,请选用Dozer)。
Dozer怎么用
使用Dozer的流程非常清楚:
- 配置你需要mapping的类,以及类之间的属性之间的对应关系。
- 根据你的配置初始化Mapper
- 使用mapper.map(source, target.class)获取转换后的对象。
Dozer提供了2种方式去配置需要mapping的类,一种是基于XML,一种是基于API,API的方式是5.3版本才开始提供的,主要是为了解决xml方式不可动态修改mapping配置的问题。下面来看一个最基础的使用Dozer的例子:
假设在一个系统中,模块A中客户的类型是User,而模块B中客户的类型是Customer, 现在模块B需要接受模块A中客户,并转换为自己模块的客户类型Customer。
public class User {
private int userId;
private String userName;
public User(int userId, String userName) {
this.userId = userId;
this.userName = userName;
}
public int getUserId() {
return userId;
}
public void setUserId(int userId) {
this.userId = userId;
}
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
}
public class Customer {
private int customerId;
private String customerName;
public int getCustomerId() {
return customerId;
}
public String getCustomerName() {
return customerName;
}
public void setCustomerId(int customerId) {
this.customerId = customerId;
}
public void setCustomerName(String customerName) {
this.customerName = customerName;
}
}
配置User和Customer属性的之间的对应关系
<?xml version="1.0" encoding="UTF-8"?>
<mappings xmlns="http://dozer.sourceforge.net"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://dozer.sourceforge.net
http://dozer.sourceforge.net/schema/beanmapping.xsd">dia
<mapping>
<class-a>com.xianlinbox.dozer.source.User</class-a>
<class-b>com.xianlinbox.dozer.destination.Customer</class-b>
<field>
<a>userId</a>
<b>customerId</b>
</field>
<field>
<a>userName</a>
<b>customerName</b>
</field>
</mapping>
</mappings>
调用Mapper获取转换后的对象:
public class DozerMapperTest {
private DozerBeanMapper mapper;
@Before
public void setUp() throws Exception {
mapper = new DozerBeanMapper();
mapper.setMappingFiles(Arrays.asList("dozer-bean-mappings.xml"));
}
@Test
public void testMapping() throws Exception {
User user = new User(1, "user1");
Customer customer = mapper.map(user, Customer.class);
assertThat(customer.getCustomerId(),equalTo(1));
assertThat(customer.getCustomerName(),equalTo("user1"));
}
}
Dozer做对象转换有什么特点
- 默认属性Mapping,即如果属性名称一样, 就不需要显示配置,Dozer会自动处理。
- 自动做类型转换,即如果两个属性名称一样,即使类型不一样,在Dozer可理解范围内,帮你处理得妥妥的。Dozer可理解的类型范围非常广,这会极大的提升程序员的生产力
- 不必担心中间的null property,遇到null property,Dozer会把对应的所有属性全部设置为null,而不会抛NullPointerExeception。
Dozer可理解的类型转换有:
• Primitive to Primitive Wrapper
• Primitive to Custom Wrapper
• Primitive Wrapper to Primitive Wrapper
• Primitive to Primitive
• Complex Type to Complex Type
• String to Primitive
• String to Primitive Wrapper
• String to Complex Type if the Complex Type contains a String constructor
• String to Map
• Collection to Collection
• Collection to Array
• Map to Complex Type
• Map to Custom Map Type
• Enum to Enum
• Each of these can be mapped to one another: java.util.Date, java.sql.Date, java.sql.Time, java.sql.Timestamp, java.util.Calendar, java.util.GregorianCalendar
• String to any of the supported Date/Calendar Objects.
• Objects containing a toString() method that produces a long representing time in (ms) to any supported Date/Calendar object.
Dozer官网地址:
http://dozer.sourceforge.net/