MapStruct 使用姿勢

背景

在代碼開發(fā)中匙头,我們通常都會使用分層架構(gòu)属拾,在分層架構(gòu)中都會使用模型轉(zhuǎn)換,在不同的層使用不同的模型胀蛮。以 DDD 分層模型為例院刁,如下:


image.png

模型分類

DO

DataObject,數(shù)據(jù)庫映射對象粪狼,通常用于基礎(chǔ)設(shè)施層退腥,與數(shù)據(jù)庫字段完全對應(yīng)任岸。

Entity

領(lǐng)域?qū)ο螅ǔS糜趹?yīng)用層和領(lǐng)域?qū)樱ㄓ幸恍?DDD 代碼模型在應(yīng)用層使用的是 DTO狡刘,但是基于應(yīng)用層是業(yè)務(wù)編排的職責(zé)享潜,可能會直接使用 Entity 的行為進(jìn)行邏輯編排,那么個人建議應(yīng)用層應(yīng)該使用 Entity)嗅蔬。不只是指實體剑按、還包括值對象。通常是充血模型澜术,包括屬性和行為吕座。

DTO

數(shù)據(jù)傳輸對象,通常用于用戶接口層(用戶接口層瘪板,通常指的是流量入口,包括web 流量漆诽、服務(wù)消費者 RPC 調(diào)用侮攀、消息輸入等)。所以 DTO 通常用于Controller中的輸入輸出參數(shù)厢拭、打到二方包里的輸入輸出參數(shù)(例如兰英,Dubbo 接口的輸入輸出參數(shù))以及消息消費者中的消息模型。
根據(jù)實際需要供鸠,有時候在 web 中畦贸,我們也會使用 vo。

轉(zhuǎn)換器

DTOAssembler

DTO 和 Entity 的轉(zhuǎn)換器

DOConverter

DO 和 Entity 的轉(zhuǎn)換器

現(xiàn)有 Bean 轉(zhuǎn)換工具的比較

目前的轉(zhuǎn)化器有:手寫轉(zhuǎn)換器楞捂、Apache BeanUtils薄坏、Spring BeanUtils、Dozer寨闹、Orika胶坠、ModelMapper、JMapper繁堡、MapStruct 等沈善。其中手寫轉(zhuǎn)換器帶來的人工成本較高,尤其是當(dāng)轉(zhuǎn)換對象屬性較多椭蹄,或者有嵌套屬性時闻牡,費時費力,且容易遺漏出錯绳矩,而且隨著對象的迭代罩润,轉(zhuǎn)換器中的代碼也要變動,所以通常我們還是會采用自動化的轉(zhuǎn)換器埋酬。
根據(jù) 這篇文章 的性能壓測來看哨啃,JMapper 和 MapStruct 的性能最好烧栋,根據(jù)易用性來講 MapStruct 最好,所以我們就使用 MapStruct 來實現(xiàn)轉(zhuǎn)換器拳球。

MapStruct 使用

    <properties>
        <org.mapstruct.version>1.4.1.Final</org.mapstruct.version>
    </properties>

    <dependencies>
        <!-- mapStruct 核心注解 -->
        <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct</artifactId>
            <version>${org.mapstruct.version}</version>
        </dependency>
        <!-- mapStruct 根據(jù)接口生成實現(xiàn)類 -->
        <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct-processor</artifactId>
            <version>${org.mapstruct.version}</version>
            <scope>provided</scope>
        </dependency>
        <!-- lombok -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>1.18.16</version>
            <scope>provided</scope>
        </dependency>
        <!-- mapStruct 支持 lombok -->
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok-mapstruct-binding</artifactId>
            <version>0.2.0</version>
            <scope>provided</scope>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.8.1</version>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                </configuration>
            </plugin>
        </plugins>
    </build>

最簡示例

DTO & Entity

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private String name;
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private String name;
}

轉(zhuǎn)換類

@Mapper
public interface Converter {
    Converter INSTANCE = Mappers.getMapper(Converter.class);
    Target fromSource(Source source);
    Source toSource(Target target);
}

測試類

public class Test {
    public static void main(String[] args) {
        testFromSource();
        testToSource();
    }

    private static void testFromSource(){
        Source source = new Source(1, "測試基礎(chǔ)轉(zhuǎn)換");
        Target target = Converter.INSTANCE.fromSource(source);
        System.out.println(target);
    }

    private static void testToSource(){
        Target target = new Target(1, "測試基礎(chǔ)轉(zhuǎn)換");
        Source source = Converter.INSTANCE.toSource(target);
        System.out.println(source);
    }
}

不同名稱的屬性關(guān)聯(lián)

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private String name; // 映射 Target 中的 targetName
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private String targetName; // 映射 Source 中的 name
}

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

    @Mapping(source = "name", target = "targetName")
    Target fromSource(Source source);

    @InheritInverseConfiguration
    Source toSource(Target target);
}
  1. 使用 @Mapping 手動映射屬性审姓;
  2. 使用 @InheritInverseConfiguration 表示繼承反方向的配置,例如祝峻,上例中的 toSource 方法的注解可以硬編碼為 @Mapping(source = "targetName", target = "name")魔吐,效果相同

不同類型的屬性關(guān)聯(lián)

@Data
@AllArgsConstructor
public class Source {
    private Integer id; // 對應(yīng) Target 的 Long id
    private String price; // 對應(yīng) Target 的 Double price
}

@Data
@AllArgsConstructor
public class Target {
    private Long id;
    private Double price;
}

@Mapper
public interface Converter {
    Converter INSTANCE = Mappers.getMapper(Converter.class);
    Target fromSource(Source source);
    Source toSource(Target target);
}

屬性名相同的屬性如果類型不同,會直接進(jìn)行類型自動轉(zhuǎn)換

內(nèi)嵌屬性關(guān)聯(lián)

@Data
@AllArgsConstructor
public class Source {
    private Integer id; // 對應(yīng) Target.TargetId.id
    private String name;
}

@Data
@AllArgsConstructor
public class Target {
    private TargetId targetId;
    private String name;
}

@Data
@AllArgsConstructor
public class TargetId {
    private Integer id;

    public static TargetId of(Integer id) {
        if (id == null) {
            throw new RuntimeException("id 不能為 null");
        }
        return new TargetId(id);
    }
}

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

    @Mapping(source = "id", target = "targetId.id")
    Target fromSource(Source source);

    @InheritInverseConfiguration
    Source toSource(Target target);
}

直接在 Mapping 中做屬性嵌套轉(zhuǎn)換

枚舉類關(guān)聯(lián)(屬性抽壤痴摇)

簡單枚舉類

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private String type;
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private SimpleEnumType type;
}

public enum SimpleEnumType {
    HAHA, HEHE
}
or
public enum SimpleEnumType {
    HAHA("HAHA"), HEHE("HEHE");

    private String desc;

    SimpleEnumType(String desc) {
        this.desc = desc;
    }
}

@Mapper
public interface Converter {
    Converter INSTANCE = Mappers.getMapper(Converter.class);
    Target fromSource(Source source);
    Source toSource(Target target);
}

簡單枚舉類:單個參數(shù)的枚舉類會自動進(jìn)行類型轉(zhuǎn)換

復(fù)雜枚舉類

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private String name; // 映射 Target.targetName
    private Integer typeCode; // 映射 Target.type.code
    private String typeName; // 映射 Target.type.name
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private String targetName;
    private ComplexEnumType type;
}

@Getter
public enum ComplexEnumType {
    HAHA(1, "haha"), HEHE(2, "hehe");

    private Integer code;
    private String name;

    ComplexEnumType(Integer code, String name) {
        this.code = code;
        this.name = name;
    }

    public static ComplexEnumType getByCode(Integer code) {
        return Arrays.stream(values()).filter(x->x.getCode().equals(code)).findFirst().orElse(null);
    }
}

Java 表達(dá)式

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

    @Mapping(source = "name", target = "targetName")
    @Mapping(target = "type", expression = "java(ComplexEnumType.getByCode(source.getTypeCode()))")
    Target fromSource(Source source);

    @InheritInverseConfiguration
    @Mapping(target = "typeCode", source = "type.code")
    @Mapping(target = "typeName", source = "type.name")
    Source toSource(Target target);
}
  1. expression:格式:java(xxx)酬姆,其中的 xxx 是 Java 語法,其計算出來的值會填充到 target 中奥溺。當(dāng) IDEA 安裝了 MapStruct Support 插件時辞色,在編寫 xxx 時會有提示。上述的 toSource 直接使用了嵌套屬性獲取方式浮定,也可以使用 @Mapping(target = "typeName", expression = "java(target.getType().getName())") 這樣的格式相满。
  2. @InheritInverseConfiguration:特殊值特殊處理,比如這里的枚舉相關(guān)值桦卒,其他屬性依舊使用逆轉(zhuǎn)繼承即可立美。

Qualifier 注解

import org.mapstruct.Qualifier;
public class ComplexEnumTypeUtil {
    @Qualifier
    @Target(ElementType.METHOD)
    @Retention(RetentionPolicy.SOURCE)
    public @interface TypeCode {
    }

    @Qualifier
    @Target(ElementType.METHOD)
    @Retention(RetentionPolicy.SOURCE)
    public @interface TypeName {
    }

    @TypeCode
    public Integer typeCode(ComplexEnumType type) {
        return type.getCode();
    }

    @TypeName
    public String typeName(ComplexEnumType type) {
        return type.getName();
    }
}

@Mapper(uses = ComplexEnumTypeUtil.class)
public interface ConverterWithQualifier {
    ConverterWithQualifier INSTANCE = Mappers.getMapper(ConverterWithQualifier.class);

    @Mapping(source = "name", target = "targetName")
    @Mapping(target = "type", expression = "java(ComplexEnumType.getByCode(source.getTypeCode()))")
    Target fromSource(Source source);

    @InheritInverseConfiguration
    @Mapping(source = "type", target = "typeCode", qualifiedBy = ComplexEnumTypeUtil.TypeCode.class)
    @Mapping(source = "type", target = "typeName", qualifiedBy = ComplexEnumTypeUtil.TypeName.class)
    Source toSource(Target target);
}

轉(zhuǎn)換類上 @Mapper(uses ={xxx.class} 可以指定使用的轉(zhuǎn)換輔助類

Name 注解

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

    @Mapping(source = "name", target = "targetName")
    @Mapping(target = "type", expression = "java(ComplexEnumType.getByCode(source.getTypeCode()))")
    Target fromSource(Source source);

    @InheritInverseConfiguration
    @Mapping(source = "type", target = "typeCode", qualifiedByName = "typeCodeUtil")
    @Mapping(source = "type", target = "typeName", qualifiedByName = "typeNameUtil")
    Source toSource(Target target);

    @Named("typeCodeUtil")
    default Integer typeCode(ComplexEnumType type) {
        return type.getCode();
    }

    @Named("typeNameUtil")
    default String typeName(ComplexEnumType type) {
        return type.getName();
    }
}

三種方式:Java Expression 最簡單,推薦使用

null 值映射時忽略或者填充默認(rèn)值

@Mapper(nullValuePropertyMappingStrategy = NullValuePropertyMappingStrategy.IGNORE)
public interface Converter {
    Converter INSTANCE = Mappers.getMapper(Converter.class);
    Target fromSource(Source source);
    Source toSource(Target target);
}

nullValuePropertyMappingStrategy 的解釋

public enum NullValuePropertyMappingStrategy {

    /**
     * If a source bean property equals {@code null} the target bean property will be set explicitly to {@code null}.
     */
    SET_TO_NULL,

    /**
     * If a source bean property equals {@code null} the target bean property will be set to its default value.
     * <p>
     * This means:
     * <ol>
     * <li>For {@code List} MapStruct generates an {@code ArrayList}</li>
     * <li>For {@code Map} a {@code HashMap}</li>
     * <li>For arrays an empty array</li>
     * <li>For {@code String} {@code ""}</li>
     * <li>for primitive / boxed types a representation of {@code 0} or {@code false}</li>
     * <li>For all other objects an new instance is created, requiring an empty constructor.</li>
     * </ol>
     * <p>
     * Make sure that a {@link Mapping#defaultValue()} is defined if no empty constructor is available on
     * the default value.
     */
    SET_TO_DEFAULT,

    /**
     * If a source bean property equals {@code null} the target bean property will be ignored and retain its
     * existing value.
     */
    IGNORE;
}

指定不映射某些值

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private String name;
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private String name;
}

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

    // name 值不做映射
    @Mapping(source = "name", target = "name", ignore = true)
    Target fromSource(Source source);

    @InheritInverseConfiguration
    Source toSource(Target target);
}

通過 @Mapping#ignore=true 來指定不需要做映射的值

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private List<SourceItem> itemList;
}

@Data
@AllArgsConstructor
public class SourceItem {
    private String identifier;
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private List<TargetItem> itemList;
}

@Data
@AllArgsConstructor
public class TargetItem {
    private String identifier;
}

@Mapper
public interface SourceItemConverter {
    SourceItemConverter INSTANCE = Mappers.getMapper(SourceItemConverter.class);
    TargetItem fromSourceItem(SourceItem sourceItem);
    SourceItem toSourceItem(TargetItem targetItem);
}

@Mapper
public interface SourceConverter {
    SourceConverter INSTANCE = Mappers.getMapper(SourceConverter.class);
    Target fromSource(Source source);
    Source toSource(Target target);
}

public class Test {
    public static void main(String[] args) {
        testFromSource();
        testToSource();
    }

    private static void testFromSource(){
        Target target = SourceConverter.INSTANCE.fromSource(new Source(1, Arrays.asList(new SourceItem("111"), new SourceItem("112"))));
        System.out.println(target);
    }

    private static void testToSource(){
        Source source = SourceConverter.INSTANCE.toSource(new Target(2, Arrays.asList(new TargetItem("222"), new TargetItem("223"))));
        System.out.println(source);
    }
}

各寫各的映射器方灾,應(yīng)用的時候是需要調(diào)用最外層的映射器即可建蹄。

更新目標(biāo)類而不是新建目標(biāo)類

@Data
@AllArgsConstructor
public class Source {
    private Integer id;
    private String name;
}

@Data
@AllArgsConstructor
public class Target {
    private Integer id;
    private String name;
}

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

    /**
     * id 不做更新,其他 source 的屬性更新到 target
     * @param source
     * @param target
     */
    @Mapping(target = "id", ignore = true)
    void fromSource(Source source, @MappingTarget Target target);
}

public class Test {
    public static void main(String[] args) {
        testFromSource();
    }

    private static void testFromSource(){
        Source source = new Source(1, "sourceName");
        Target target = new Target(2, "targetName");
        Converter.INSTANCE.fromSource(source, target);
        System.out.println(target);
    }
}

MapStruct 原理

以上述的最簡示例為例裕偿,在項目編譯時洞慎,會把如下轉(zhuǎn)換接口動態(tài)編譯出實現(xiàn)類(底層使用了 APT 技術(shù),APT 示例見這里)击费。實現(xiàn)類與手寫的轉(zhuǎn)換器類似拢蛋,使用構(gòu)造器或者 setter/getter 進(jìn)行操作。
在運行時蔫巩,直接執(zhí)行該實現(xiàn)類谆棱,所以性能與手寫幾乎相同。

@Mapper
public interface Converter {
    Converter INSTANCE = Mappers.getMapper(Converter.class);
    Target fromSource(Source source);
    Source toSource(Target target);
}

其實現(xiàn)類如下:

package xxx; // 與接口所在的包名相同

import javax.annotation.Generated;

@Generated(
    value = "org.mapstruct.ap.MappingProcessor",
    date = "2021-01-21T21:19:02+0800",
    comments = "version: 1.4.1.Final, compiler: javac, environment: Java 1.8.0_151 (Oracle Corporation)"
)
public class ConverterImpl implements Converter {

    @Override
    public Target fromSource(Source source) {
        if ( source == null ) {
            return null;
        }

        Integer id = null;
        String name = null;

        id = source.getId();
        name = source.getName();

        Target target = new Target( id, name );

        return target;
    }

    @Override
    public Source toSource(Target target) {
        if ( target == null ) {
            return null;
        }

        Integer id = null;
        String name = null;

        id = target.getId();
        name = target.getName();

        Source source = new Source( id, name );

        return source;
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末圆仔,一起剝皮案震驚了整個濱河市垃瞧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌坪郭,老刑警劉巖个从,帶你破解...
    沈念sama閱讀 218,386評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡嗦锐,警方通過查閱死者的電腦和手機嫌松,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評論 3 394
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來奕污,“玉大人萎羔,你說我怎么就攤上這事√寄” “怎么了贾陷?”我有些...
    開封第一講書人閱讀 164,704評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長嘱根。 經(jīng)常有香客問我髓废,道長,這世上最難降的妖魔是什么该抒? 我笑而不...
    開封第一講書人閱讀 58,702評論 1 294
  • 正文 為了忘掉前任慌洪,我火速辦了婚禮,結(jié)果婚禮上凑保,老公的妹妹穿的比我還像新娘蒋譬。我一直安慰自己,他們只是感情好愉适,可當(dāng)我...
    茶點故事閱讀 67,716評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著癣漆,像睡著了一般维咸。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上惠爽,一...
    開封第一講書人閱讀 51,573評論 1 305
  • 那天癌蓖,我揣著相機與錄音,去河邊找鬼婚肆。 笑死租副,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的较性。 我是一名探鬼主播用僧,決...
    沈念sama閱讀 40,314評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼赞咙!你這毒婦竟也來了责循?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,230評論 0 276
  • 序言:老撾萬榮一對情侶失蹤攀操,失蹤者是張志新(化名)和其女友劉穎院仿,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,680評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡歹垫,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,873評論 3 336
  • 正文 我和宋清朗相戀三年剥汤,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片排惨。...
    茶點故事閱讀 39,991評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡吭敢,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出若贮,到底是詐尸還是另有隱情省有,我是刑警寧澤,帶...
    沈念sama閱讀 35,706評論 5 346
  • 正文 年R本政府宣布谴麦,位于F島的核電站蠢沿,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏匾效。R本人自食惡果不足惜舷蟀,卻給世界環(huán)境...
    茶點故事閱讀 41,329評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望面哼。 院中可真熱鬧野宜,春花似錦、人聲如沸魔策。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽闯袒。三九已至虎敦,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間政敢,已是汗流浹背其徙。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留喷户,地道東北人唾那。 一個月前我還...
    沈念sama閱讀 48,158評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像褪尝,于是被迫代替她去往敵國和親闹获。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,941評論 2 355

推薦閱讀更多精彩內(nèi)容

  • MapStruct是一種類型安全的bean映射類生成java注釋處理器河哑。我們要做的就是定義一個映射器接口昌罩,聲明任何...
    恒宇少年閱讀 90,599評論 21 123
  • 介紹 在我們開發(fā)中,涉及到對各種DO灾馒,VO茎用,DTO之間的轉(zhuǎn)換,如果你還在使用下面的工具類做這些工作 那么我覺得你很...
    土豆肉絲蓋澆飯閱讀 2,287評論 0 10
  • 日常開發(fā)中,我們時長會寫很多關(guān)于PO轉(zhuǎn)VO的代碼或者是VO轉(zhuǎn)DTO相關(guān)的代碼轨功,造成我們的程序異常的臃腫旭斥。如下: 編...
    茶還是咖啡閱讀 23,210評論 0 10
  • 推薦指數(shù): 6.0 書籍主旨關(guān)鍵詞:特權(quán)、焦點古涧、注意力垂券、語言聯(lián)想、情景聯(lián)想 觀點: 1.統(tǒng)計學(xué)現(xiàn)在叫數(shù)據(jù)分析羡滑,社會...
    Jenaral閱讀 5,721評論 0 5
  • 昨天菇爪,在回家的路上,坐在車?yán)镉圃沼圃盏乜粗摹度龉衬墓适隆菲饣瑁冶焕锩娴膬?nèi)容深深吸引住了凳宙,盡管上學(xué)時...
    夜闌曉語閱讀 3,788評論 2 9