真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

秒懂Java之實(shí)體轉(zhuǎn)化利器MapStruct詳解-創(chuàng)新互聯(lián)

[版權(quán)申明]非商業(yè)目的注明出處可自由轉(zhuǎn)載
出自:shusheng007

十多年專注成都網(wǎng)站制作,企業(yè)網(wǎng)站制作,個人網(wǎng)站制作服務(wù),為大家分享網(wǎng)站制作知識、方案,網(wǎng)站設(shè)計(jì)流程、步驟,成功服務(wù)上千家企業(yè)。為您提供網(wǎng)站建設(shè),網(wǎng)站制作,網(wǎng)頁設(shè)計(jì)及定制高端網(wǎng)站建設(shè)服務(wù),專注于企業(yè)網(wǎng)站制作,高端網(wǎng)頁制作,對小攪拌車等多個行業(yè),擁有豐富的網(wǎng)站設(shè)計(jì)經(jīng)驗(yàn)。

文章目錄
  • 概述
  • MapStruct
    • 簡介:
    • 使用
      • 如何配置
    • 應(yīng)用
    • 進(jìn)階
      • 裝插件
      • 調(diào)用方式
      • 自定義映射
      • 忽略映射
      • 設(shè)置默認(rèn)值
      • 設(shè)置常量
      • 數(shù)據(jù)類型轉(zhuǎn)換
      • 表達(dá)式
      • 嵌套映射
      • 集合映射
      • 外部引用
      • 多個數(shù)據(jù)源
      • 切面操作
  • 總結(jié)
  • 源碼

概述

由于現(xiàn)代程序在追求擴(kuò)展和維護(hù)性時很多采用分層的設(shè)計(jì)結(jié)構(gòu),所以在寫程序時候需要在各種實(shí)體之間互相轉(zhuǎn)換,而他們之間很多時候在業(yè)務(wù)或者技術(shù)架構(gòu)上區(qū)別較大,在具體的屬性上差別卻很小。

例如將Programer轉(zhuǎn)換為ProgramerDto就很普遍,如下所示:

public class Programer {
    private String name;
    private String proLang;
}

轉(zhuǎn)換為:

public class ProgramerDto {
    private String name;
    private String proLang;
}

由于這些是繁瑣易錯且沒有技術(shù)含量的編碼工作,所以聰明的程序員就會尋求不斷簡化它的方法,MapStruct就是其中的一個利器。

MapStruct 簡介:

MapStruct is a Java annotation processor for the generation of type-safe and performant mappers for Java bean classes

大意就是:MapStruct 是一個用于Java的Bean的映射器,是它是基于注解的,而且是編譯時APT(annotation processor tool)。不像其他APT是運(yùn)行時,例如Spring里面的注解處理方式,是在運(yùn)行時通過反射的方式處理的。

詳細(xì)介紹可以到其官網(wǎng)查看:MapStruct源碼,下面是官方給出的選擇MapStruc的理由,你看看是否說服了你去使用它:

  • Fast execution by using plain method invocations instead of reflection
  • Compile-time type safety. Only objects and attributes mapping to each other can be mapped, so there’s no accidental mapping of an order entity into a customer DTO, etc.
  • Self-contained code—no runtime dependencies
  • Clear error reports at build time if:
    • mappings are incomplete (not all target properties are mapped)
    • mappings are incorrect (cannot find a proper mapping method or type conversion)
  • Easily debuggable mapping code (or editable by hand—e.g. in case of a bug in the generator)
使用 如何配置

從前面的介紹我們得知,MapStruct是通過在編譯時通過注解來生成代碼的方式工作的,所以需要配置APT。此處我們還想使用lombok,所以也會順便配置其與lombok結(jié)合的配置。

UTF-81.81.81.5.3.Final1.18.200.2.0org.mapstructmapstruct${mapstruct.version}org.projectlomboklombok${lombok.version}provided		...
      org.apache.maven.pluginsmaven-compiler-plugin 3.8.1  ${maven.compiler.source} ${maven.compiler.target}
                  org.mapstructmapstruct-processor ${mapstruct.version}   org.projectlomboklombok ${lombok.version}   org.projectlomboklombok-mapstruct-binding ${lombok-mapstruct-binding.version}    

如上所示,主要配置了注解處理器:。如果不使用lombok的話,去掉相應(yīng)的配置即可。

應(yīng)用

當(dāng)完成了配置就就需要寫代碼了,主要是一些注解的使用,MapStruc提供的功能是很強(qiáng)大的,但是入門很容易的。

假設(shè)我們有如下兩個需要轉(zhuǎn)換的類:

@Data
public class Programer {
    private String name;
    private String lang;
    private Double height;
    private Date beDate;
    private Address address;
    private String girlName;
    private String girlDes;
}

@Data
public class ProgramerDto {
    private String name;
    private String proLang;
    private String height;
    private String beDate;
    private AddressDto address;
    private GirlFriendDto girlFriend;
}

第一步: 定義一個interface,使用@Mapper標(biāo)記

@Mapper
public interface ProgramerConvetor {
...
}

第二步:構(gòu)建一個實(shí)例屬性用于訪問里面的方法。

@Mapper
public interface ProgramerConvetor {
    ProgramerConvetor INSTANCE = Mappers.getMapper(ProgramerConvetor.class);
}

第三步:提供轉(zhuǎn)換方法申明,必要時使用@Mapping注解

@Mapper
public interface ProgramerConvetor {
    ProgramerConvetor INSTANCE = Mappers.getMapper(ProgramerConvetor.class);

    @Mapping(target = "lang", source = "proLang")
    ProgramerDto toProgramerDto(Programer programer);
}

MapStruc默認(rèn)會將兩個bean的名稱相同的屬性進(jìn)行映射,如果source與target的屬性名稱不一致則需要借助@Mapping注解。

簡單的轉(zhuǎn)換就只需要以上3步就可以了,編譯程序后就會在\target\generated-sources\annotations下產(chǎn)生實(shí)現(xiàn)類了。

下面的代碼是MapStruc自動生成的:

@Generated(
    value = "org.mapstruct.ap.MappingProcessor",
    date = "2023-01-08T16:51:05+0800",
    comments = "version: 1.5.3.Final, compiler: javac, environment: Java 11.0.16.1 (Oracle Corporation)"
)
public class ProgramerConvetorImpl implements ProgramerConvetor {

    @Override
    public ProgramerDto toProgramerDto(Programer programer) {
        if ( programer == null ) {
            return null;
        }

        ProgramerDto programerDto = new ProgramerDto();
        programerDto.setLang( programer.getProLang() );
        programerDto.setName( programer.getName() );
        ...
        return programerDto;
    }
}

是不是和你手寫的也差不多,那有了生成類我們就可以在代碼中使用了:

public void runMap(){
    Programer programer = new Programer();
    programer.setName("shusheng007");
    ...
    
    ProgramerDto programerDto = ProgramerConvetor.INSTANCE.toProgramerDto(programer);
    log.info("dto: {}",programerDto);
}

可見,可以通過轉(zhuǎn)換器接口里面的那個INSTANCE實(shí)例屬性來調(diào)用其方法。看是不是比你手寫方便多了呢?特別是屬性比較多,而其名稱又有很多一致的情況下就更方便了。

進(jìn)階

前面那個是最基礎(chǔ)的使用,MapStruc提供了非常靈活的映射方式,要完全掌握既沒有必要又是不可能的,下面我們挑幾個常用的以應(yīng)對80%的日常工作。

裝插件

工欲善其事必先利其器,咱給Idea裝上一個插件 MapStruct Support,各種代碼智能提示走起來…

調(diào)用方式

前面我們使用在接口中定義一個實(shí)例屬性的方式來訪問生成的方法,這有點(diǎn)不Spring,在Spring中我們習(xí)慣將bean交給Spring容器管理,MapSturc也支持。

@Mapper(componentModel = MappingConstants.ComponentModel.SPRING)
public interface ProgramerConvetor {
}

通過修改@MappercomponentModel的屬性為spring即可。下面是生成的代碼,發(fā)現(xiàn)已經(jīng)添加了@Component注解。

...
@Component
public class ProgramerConvetorImpl implements ProgramerConveto{
}
自定義映射

當(dāng)source與target里的屬性名稱不一致時需要顯示指定映射關(guān)系

@Mapping(target = "lang", source = "proLang")
ProgramerDto toProgramerDto(Programer programer);

生成代碼:

ProgramerDto programerDto = new ProgramerDto();
programerDto.setLang( programer.getProLang() );
忽略映射

如果不想給ProgramerDtoproLang賦值可以忽略它。

@Mapping(target = "proLang", ignore = true)
ProgramerDto toProgramerDto(Programer programer);
設(shè)置默認(rèn)值

如果想實(shí)現(xiàn)在source值為null時給一個默認(rèn)值也是可以了。

@Mapping(target = "proLang", defaultValue = "java")
ProgramerDto toProgramerDto(Programer programer);

生成代碼:

ProgramerDto programerDto = new ProgramerDto();
if ( programer.getProLang() != null ) {
    programerDto.setProLang( programer.getProLang() );
}
else {
    programerDto.setProLang( "java" );
}

其實(shí)默認(rèn)值不僅可以是一個具體的值,還可以是一個表達(dá)式,表達(dá)式一會我們再說。

設(shè)置常量

給source的某個屬性賦值為常量

@Mapping(target = "proLang", constant = "kotlin")
ProgramerDto toProgramerDto(Programer programer);

生成代碼:

programerDto.setProLang( "kotlin" );
數(shù)據(jù)類型轉(zhuǎn)換

我們在進(jìn)行bean映射的時候,有時會遇到數(shù)據(jù)類型不一致的情況。例如對于一個日期,source的數(shù)據(jù)類型是Date,而target的數(shù)據(jù)類型是String,這些情況怎么處理呢?

public class Programer {
    private Double height;
    private Date beDate;
}
public class ProgramerDto {
    private String height;
    private String beDate;
}

從上面的代碼可以看到,我們的兩個bean的數(shù)據(jù)類型是不一致的,但是MapStruct卻可以幫我們自動轉(zhuǎn)換

@Mapping(target = "height", source = "height")
ProgramerDto toProgramerDto(Programer programer);

生成的代碼:

if ( programer.getHeight() != null ) {
    programerDto.setHeight( String.valueOf( programer.getHeight() ) );
}

生成的代碼已經(jīng)將Double幫我們轉(zhuǎn)換成String了。不僅如此,我們還可以對生成的字符串的格式進(jìn)行設(shè)置,例如將身高數(shù)據(jù)保留兩位小數(shù)

@Mapping(target = "height", source = "height" ,numberFormat = "#.00")

生成的代碼:

if ( programer.getHeight() != null ) {
    programerDto.setHeight( new DecimalFormat( "#.00" ).format( programer.getHeight() ) );
}

下面看一個日期相關(guān)的轉(zhuǎn)換

@Mapping(target = "beDate", dateFormat = "yyyy-MM-dd HH:mm:ss")
ProgramerDto toProgramerDto(Programer programer);

生成的代碼:

if ( programer.getBeDate() != null ) {
    programerDto.setBeDate( new SimpleDateFormat( "yyyy-MM-dd HH:mm:ss" ).format( programer.getBeDate() ) );
}

可見在日期轉(zhuǎn)換中我們可以控制器轉(zhuǎn)換后的格式。

表達(dá)式

這個就比較厲害了,你就認(rèn)為是方法調(diào)用即可,我喜歡。有時我們會遇到在做映射的時候不是簡單的賦值,而是要進(jìn)行計(jì)算,那這個功能就可以使用表達(dá)式來完成。

例如我們要實(shí)現(xiàn)將程序員名稱變成大寫個功能,就可以使用expression這個屬性進(jìn)行配置。表達(dá)式的形式如下:java(代碼調(diào)用)。

@Mapping(target = "name", expression = "java(programer.getName().toUpperCase())")
ProgramerDto toProgramerDto(Programer programer);

生成代碼:

programerDto.setName( programer.getName().toUpperCase() );

表達(dá)式里可以進(jìn)行方法調(diào)用,例如上面的代碼我們可以換一種方式寫,將轉(zhuǎn)換代碼寫成一個default函數(shù)。

注意這個defalut方法的簽名一定要符合你的需求,因?yàn)镸S會為每一個映射嘗試這個方法,一旦符合了就會被使用,例如你寫一個String 到 String的轉(zhuǎn)換那就麻煩了,每個符合這個得屬性轉(zhuǎn)換都會用上…

@Mapping(target = "name", expression = "java(nameToUp(programer))")
ProgramerDto toProgramerDto(Programer programer);

default String nameToUp(Programer programer){
    return Optional.ofNullable(programer)
            .filter(Objects::nonNull)
            .map(p->p.getName())
            .orElse(null)
            .toUpperCase();
}

對于分不清的情況所可以使用和Spring類似的方案,就是使用qualified。例如上面的功能還可以以下面的方案實(shí)現(xiàn)。

@Mapping(target = "name", qualifiedByName ={"nameToUp"} )
ProgramerDto toProgramerDto(Programer programer);

@Named("nameToUp")
default String nameToUp(String name) {
    return name.toUpperCase();
}
嵌套映射

我們經(jīng)常會遇到bean里面套著bean的映射。

{
    "name":"shusheng007",
    "address":{
        "country":"China",
        "city":"TianJin"
    }
}

對于這樣的映射,我們只需要在mapper中提供一個嵌套bean的轉(zhuǎn)換關(guān)系即可。

@Mapping(target = "address", source = "address")
ProgramerDto toProgramerDto(Programer programer);
//嵌套bean的轉(zhuǎn)換關(guān)系
AddressDto toAddressDto(Address addr);

生成代碼:

programerDto.setAddress( toAddressDto( programer.getAddress() ) );

@Override
public AddressDto toAddressDto(Address addr) {
    if ( addr == null ) {
        return null;
    }

    AddressDto addressDto = new AddressDto();

    addressDto.setCountry( addr.getCountry() );
    addressDto.setCity( addr.getCity() );

    return addressDto;
}

其實(shí)MapStruct非常智能的,即使你不提供它也會嘗試進(jìn)行映射的。

集合映射

只需要提供集合元素類型的映射即可。

AddressDto toAddressDto(Address addr);

List toAddressList(List addrList);

生成代碼:

@Override
public AddressDto toAddressDto(Address addr) {
    ...
}

@Override
public List toAddressList(List addrList) {
    if ( addrList == null ) {
        return null;
    }

    List list = new ArrayList( addrList.size() );
    for ( Address address : addrList ) {
        list.add( toAddressDto( address ) );
    }

    return list;
}
外部引用

上面我們介紹了表達(dá)式,通過它我們可以寫代碼邏輯,但是當(dāng)轉(zhuǎn)換關(guān)系需要調(diào)用外部類的方法時怎么辦呢?我們有兩種方法,下面看一下。

例如我們有如下要被引用的類:

@Component
public class GirlFriendMapper {

    public GirlFriendDto toGirlFriendDto(Programer programer) {
        GirlFriendDto girlFriendDto = new GirlFriendDto();
        girlFriendDto.setName(programer.getName());
        girlFriendDto.setDescription(programer.getGirlDes());
        return girlFriendDto;
    }
}
  • 使用抽象類代替接口來做Mapper
@Mapper(componentModel = MappingConstants.ComponentModel.SPRING)
public abstract class ClzProgramerConvertor {
    @Autowired
    protected GirlFriendMapper girlFriendMapper;
    ...
    @Mapping(target = "girlFriend", expression = "java(girlFriendMapper.toGirlFriendDto(programer))")
    public abstract ProgramerDto toProgramerDto(Programer programer);
}

使用了抽象類后,你發(fā)現(xiàn)熟悉的味道回來了,可以使用@Autowired隨便往里面注入實(shí)例了,然后在expression里面調(diào)用就好了,是不是很爽?

生成代碼:

@Generated(
    value = "org.mapstruct.ap.MappingProcessor",
    date = "2023-01-08T21:03:05+0800",
    comments = "version: 1.5.3.Final, compiler: javac, environment: Java 11.0.16.1 (Oracle Corporation)"
)
@Component
public class ClzProgramerConvertorImpl extends ClzProgramerConvertor {

    @Override
    public ProgramerDto toProgramerDto(Programer programer) {
        ProgramerDto programerDto = new ProgramerDto();
		...
        programerDto.setGirlFriend( girlFriendMapper.toGirlFriendDto(programer) );

        return programerDto;
    }

    protected AddressDto addressToAddressDto(Address address) {
     ...
    }
}

那個girlFriendMapper就是我們在父類中注入的。

  • 使用@Mapper注解的import和use屬性

import屬性就和java中的import是一樣的,導(dǎo)入后在expression中就可以不使用類的全限定名稱了。例如你的轉(zhuǎn)換用到了一個靜態(tài)工具類,那么如果不在import中導(dǎo)入此工具類,那么使用的時候就要全限定名了。

@Mapping(target = "name", expression = "java(top.ss007.Util.toUpper(programer.getName()))")

當(dāng)使用的映射方法在其他非靜態(tài)類里時,就可以使用use屬性。

@Mapper(componentModel = MappingConstants.ComponentModel.SPRING,
        uses = {
            GirlFriendMapper.class
        }
)
public interface ProgramerConvetor {
    @Mapping(target = "girlFriend", source = "programer")
    ProgramerDto toProgramerDto(Programer programer);
}

我們使用@Mapperuse屬性將GirlFriendMapper引入進(jìn)來。

@Generated(
    value = "org.mapstruct.ap.MappingProcessor",
    date = "2023-01-08T21:03:17+0800",
    comments = "version: 1.5.3.Final, compiler: javac, environment: Java 11.0.16.1 (Oracle Corporation)"
)
@Component
public class ProgramerConvetorImpl implements ProgramerConvetor {
     @Autowired
    private GirlFriendMapper girlFriendMapper;

    @Override
    public ProgramerDto toProgramerDto(Programer programer) {
        ProgramerDto programerDto = new ProgramerDto();
        ...
        programerDto.setGirlFriend( girlFriendMapper.toGirlFriendDto( programer ) );
        return programerDto;
    }
 }

可見,引入的類的實(shí)例在實(shí)現(xiàn)類中被注入了。我們還可以通過injectionStrategy = InjectionStrategy.CONSTRUCTOR指定通過構(gòu)造函數(shù)來注入實(shí)例,如果不指定默認(rèn)使用屬性注入。

多個數(shù)據(jù)源

有時我們會遇到多個bean轉(zhuǎn)一個bean的情況,需顯示指定參數(shù)名稱

@Mapping(target = "name", source = "programer.name")
@Mapping(target = "girlFriendName", source = "girl.name")
ProgramerDto toProgramerDto(Programer programer, Gir girl);
切面操作

MapStruct 還提供了兩個注解@BeforeMapping, @AfterMapping用來實(shí)現(xiàn)在mapping前后的統(tǒng)一操作,這一般比較少用,但是在使用多態(tài)的時候還是很有作用的。

需求:我們有一個Human父類,有男人和女人兩個子類,然后我們要將這兩個子類型mapping成HumanDto。HumanDto中有個性別的屬性,需要根據(jù)具體的類型決定。在mapping完成后,我們還要將名稱修飾一下。

public class Human {
    private String name;
}
public class Man extends Human{
}
public class Woman extends Human{
}

public class HumanDto {
    private String name;
    private GenderType genderType;
}
public enum GenderType {
    MAN,WOMAN
}

當(dāng)然我們可以寫兩個轉(zhuǎn)換方法即可,一個Man到HumanDto,一個Woman到HumanDto,但是在使用的時候就比較麻煩了,需要傳入具體的類型,代碼也有重復(fù)。這種場景下我們就可以使用這兩個注解完美的解決問題。

@Mapper(componentModel = MappingConstants.ComponentModel.SPRING)
public abstract class HumanConvertor {

    @BeforeMapping
    protected void humanDtoWithGender(Human human, @MappingTarget HumanDto humanDto) {
        if (human instanceof Man) {
            humanDto.setGenderType(GenderType.MAN);
        } else if (human instanceof Woman) {
            humanDto.setGenderType(GenderType.WOMAN);
        }
    }

    @AfterMapping
    protected void decorateName(@MappingTarget HumanDto humanDto) {
        humanDto.setName(String.format("【%s】", humanDto.getName()));
    }

    public abstract HumanDto toHumanDto(Human human);
}

生成的代碼:

@Generated(
    value = "org.mapstruct.ap.MappingProcessor",
    date = "2023-01-09T23:45:44+0800",
    comments = "version: 1.5.3.Final, compiler: javac, environment: Java 11.0.16.1 (Oracle Corporation)"
)
@Component
public class HumanConvertorImpl extends HumanConvertor {

    @Override
    public HumanDto toHumanDto(Human human) {
        ...
        HumanDto humanDto = new HumanDto();
        //mapping前執(zhí)行
        humanDtoWithGender( human, humanDto );
        humanDto.setName( human.getName() );
        //mapping后執(zhí)行
        decorateName( humanDto );

        return humanDto;
    }
}

使用:

@Autowired
private HumanConvertor humanConvertor;

public void runHumanDemo(){
    Human man = new Man();
    man.setName("王二狗");
    log.info("{}是大男人",humanConvertor.toHumanDto(man));

    Human woman = new Woman();
    woman.setName("牛翠華");
    log.info("{}是小女人", humanConvertor.toHumanDto(woman));
}
總結(jié)

至此,MapStruct的基本操作基本上都涉及到了,足以應(yīng)對日常工作了,但是我還是那句話:那年我雙手插兜,不知道什么叫… , 哎呀我去,跑偏了。 我還是那句話:MapStruct提供了大量的注解和自定義配置,如遇到特殊需求還需要去查看官方文檔和示例。

忽忽悠悠又要過年啦,今年終于可以回老家過年了, 3年疫情終于要結(jié)束了,如果惡毒生活‘’強(qiáng)奸”了你,在反抗不了的情況下還是要學(xué)會隱忍和享受,保持樂觀的心態(tài),不斷學(xué)習(xí)以待反抗之時,去追求美好生活…

源碼

源碼請到首發(fā)文末查看:MapSturct

你是否還在尋找穩(wěn)定的海外服務(wù)器提供商?創(chuàng)新互聯(lián)www.cdcxhl.cn海外機(jī)房具備T級流量清洗系統(tǒng)配攻擊溯源,準(zhǔn)確流量調(diào)度確保服務(wù)器高可用性,企業(yè)級服務(wù)器適合批量采購,新人活動首月15元起,快前往官網(wǎng)查看詳情吧


文章標(biāo)題:秒懂Java之實(shí)體轉(zhuǎn)化利器MapStruct詳解-創(chuàng)新互聯(lián)
網(wǎng)頁地址:http://weahome.cn/article/dgjsge.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部