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

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

    1、新增Map到Java bean的转换(Mapping from Map to Bean)

    如果有以下Java Bean

    1. public class Customer {
    2. private Long id;
    3. private String name;
    4. //getters and setter omitted for brevity
    5. }

    相应的MapStruct代码如下:

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

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

    1. // GENERATED CODE
    2. public class CustomerMapperImpl implements CustomerMapper {
    3. @Override
    4. public Customer toCustomer(Map<String, String> map) {
    5. // ...
    6. if ( map.containsKey( "id" ) ) {
    7. customer.setId( Integer.parseInt( map.get( "id" ) ) );
    8. }
    9. if ( map.containsKey( "customerName" ) ) {
    10. customer.setName( source.get( "customerName" ) );
    11. }
    12. // ...
    13. }
    14. }

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

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

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

    1. @Mapper
    2. public interface CarMapper {
    3. CarDto carToCarDto(Car car);
    4. @Condition
    5. default boolean isNotEmpty(String value) {
    6. return value != null && !value.isEmpty();
    7. }
    8. }

    则Map Struct 1.5.0 Final生成的代码是:

    1. // GENERATED CODE
    2. public class CarMapperImpl implements CarMapper {
    3. @Override
    4. public CarDto carToCarDto(Car car) {
    5. if ( car == null ) {
    6. return null;
    7. }
    8. CarDto carDto = new CarDto();
    9. if ( isNotEmpty( car.getOwner() ) ) {
    10. carDto.setOwner( car.getOwner() );
    11. }
    12. // Mapping of other properties
    13. return carDto;
    14. }
    15. }

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

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

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

    1. @Mapper
    2. public interface FruitMapper {
    3. @SubclassMapping( source = AppleDto.class, target = Apple.class )
    4. @SubclassMapping( source = BananaDto.class, target = Banana.class )
    5. Fruit map( FruitDto source );
    6. }

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