MapStruct 1.5 发布,终于支持Map转为Bean

本月三号,MapStruct 1.5.0 Final发布,本次正式版距离上次正式版发布已经过去了快7年(上个正式版发布于2015年11月),此次发布除了修复了110多个bug外,还有以下新特性值得关注:

  • 支持了Map<String,?>bean的转换
  • 支持更加完备的条件转换(Conditional mapping)
  • 支持子类之间的转换(Support for subclass mapping)

1. 新增MapJava bean的转换(Mapping from Map to Bean)

如果我们有以下Java Bean

public class Customer {
    private Long id;
    private String name;
    //getters and setter omitted for brevity
}

相应的MapStruct代码如下:

@Mapper
public interface CustomerMapper {
    @Mapping(target = "name", source = "customerName")
    Customer toCustomer(Map<String, String> map);
}

那最终会生成类似如下的转换代码:

// GENERATED CODE
public class CustomerMapperImpl implements CustomerMapper {
 
    @Override
    public Customer toCustomer(Map<String, String> map) {
        // ...
        if ( map.containsKey( "id" ) ) {
            customer.setId( Integer.parseInt( map.get( "id" ) ) );
        }
        if ( map.containsKey( "customerName" ) ) {
            customer.setName( source.get( "customerName" ) );
        }
        // ...
    }
}

不过需要注意,待转换的Mapkey必须是String类型的,否则,转换代码会跳过这个key

2. 更加完备的条件转换( Conditional Mapping

1.5.0 Final版本之前,如果Java bean中含有hasXXX或者isXXX的这类方法(XXXbean中的属性名),则MapStruct生成的代码中则会调用这类方法来判断是否在转换后的bean中是否包含原来的属性,但是遗憾的是,大多数情况下,我们并不能直接修改原bean的代码。基于此,1.5.0 Final版本引入了org.mapstruct.Condition注解来实现条件转换。例如我们有如下转换代码:

@Mapper
public interface CarMapper {
 
    CarDto carToCarDto(Car car);
 
    @Condition
    default boolean isNotEmpty(String value) {
        return value != null && !value.isEmpty();
    }
}

Map Struct 1.5.0 Final生成的代码是:

// GENERATED CODE
public class CarMapperImpl implements CarMapper {
 
    @Override
    public CarDto carToCarDto(Car car) {
        if ( car == null ) {
            return null;
        }
        CarDto carDto = new CarDto();
        if ( isNotEmpty( car.getOwner() ) ) {
            carDto.setOwner( car.getOwner() );
        }
        // Mapping of other properties
        return carDto;
    }
}

org.mapstruct.Condition除了作用到整个bean外还可以修饰具体的属性值,实现bean属性维度的条件转换。

3. 增加对子类转换的支持(Subclass Mapping)

假如有父类Fruit和两个子类AppleBanana,在新特性的支持下我们的转换代码可以写的更加简洁:

@Mapper
public interface FruitMapper {

    @SubclassMapping( source = AppleDto.classtarget = Apple.class )
    @SubclassMappingsource 
= BananaDto.classtarget = Banana.class )
    Fruit mapFruitDto source )
;
}

如果Fruit是抽象类或者是接口,则会报编译错误。


往期推荐:

1、包装类这颗语法糖,其实并不甜

2、你真的了解时间吗?

3、细数ThreadLocal三大坑,内存泄露仅是小儿科

4、Java 8 ConcurrentHashMap源码中竟然隐藏着两个BUG

5、ConcurrentHashMap中有十个提升性能的细节,你都知道吗?


原文始发于微信公众号(苦味代码):MapStruct 1.5 发布,终于支持Map转为Bean

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由半码博客整理,本文链接:https://www.bmabk.com/index.php/post/21473.html

(0)
小半的头像小半

相关推荐

发表回复

登录后才能评论
半码博客——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!