移形換影-MapStruct使用技巧

介紹

在我們開發(fā)中轻绞,涉及到對各種DO采记,VO,DTO之間的轉(zhuǎn)換政勃,如果你還在使用下面的工具類做這些工作

SuppliersDTO suppliersDTO = BeanUtils.copyProperties(suppliersDO, SuppliersDTO.class);

那么我覺得你很有必要了解下我即將介紹的這個(gè)框架

競品分析

在阿里編碼規(guī)范插件中有這么一條


下面我們就來分析下 Apache BeanUtils, Spring BeanUtils唧龄,Cglib BeanCopier和本文介紹的MapStruct的差別。

框架/工具類 原理 性能 功能豐富性
Apache BeanUtils 反射 一般
Spring BeanUtils 反射
Cglib BeanCopier 字節(jié)碼生成 強(qiáng) 一般
MapStruct 字節(jié)碼生成 較強(qiáng) 強(qiáng)

為什么反射性能差奸远?簡單的講下既棺,你方法直接調(diào)用,在類加載(解析階段懒叛,符號引用替換為直接引用)的時(shí)候就知道調(diào)用的方法地址在哪里丸冕。而反射的話,運(yùn)行時(shí)獲取地址薛窥,如果一個(gè)類有10個(gè)方法胖烛,遍歷10個(gè)方法去計(jì)算出這個(gè)地址,肯定增加了耗時(shí)诅迷。

功能性的話佩番,我看除了MapStruct框架外,其他幾種都是讓你自己實(shí)現(xiàn)一些轉(zhuǎn)換器傳入罢杉,太雞肋了

以下是上面4中框架/工具類執(zhí)行100萬次對象拷貝的時(shí)間對比(單位納秒)


Cglib BeanCopier排名第一趟畏,MapStruct排名第二。

為啥同是字節(jié)碼滩租,Cglib BeanCopier比MapStruct強(qiáng)赋秀,因?yàn)槲业臏y試用例是太簡單了,那么比較下復(fù)雜的場景吧律想,不好意思猎莲,Cglib BeanCopier能夠支持的場景不夠復(fù)雜。

性能和Cglib BeanCopier上差距不大蜘欲,但是MapStruct使用起來太方便簡潔了益眉,所以我還是推薦使用MapStruct。

對比代碼見https://github.com/shengchaojie/java_framework_contrast
下面的demo代碼也在這個(gè)github項(xiàng)目中

如何使用

配置

...
<properties>
    <org.mapstruct.version>1.3.0.Final</org.mapstruct.version>
    <org.projectlombok.version>1.18.0</org.projectlombok.version>
</properties>
...
<dependencies>
    <dependency>
        <groupId>org.mapstruct</groupId>
        <artifactId>mapstruct</artifactId>
        <version>${org.mapstruct.version}</version>
    </dependency>

    <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <version>${org.projectlombok.version}</version>
            <scope>provided</scope>
        </dependency>
</dependencies>
...
<build>
    <pluginManagement>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.6.2</version>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                    <annotationProcessorPaths>
                        <path>
                            <groupId>org.mapstruct</groupId>
                            <artifactId>mapstruct-processor</artifactId>
                            <version>${org.mapstruct.version}</version>
                        </path>
                        <path>
                            <groupId>org.projectlombok</groupId>
                            <artifactId>lombok</artifactId>
                            <version>${org.projectlombok.version}</version>
                        </path>
                    </annotationProcessorPaths>
                </configuration>
            </plugin>
        </plugins>
    </pluginManagement>
</build>
...

上面的配置是和lombok集成的配置方式姥份,單不僅限以上一種配置方式

使用方式

比如我有以下兩個(gè)模型需要轉(zhuǎn)換

@Data
public class OrderDO {

    private String orderCode;

    private String address;

}
@Data
public class OrderDTO {

    private String orderCode;

    private String address;

    private List<OrderWareDetailDTO> orderWareDetailDTOS;

}

我只需要?jiǎng)?chuàng)建一個(gè)ConvertUtil郭脂,并且加上@Mapper注解即可

@Mapper
public interface ConvertUtil {

    ConvertUtil INSTANCE = Mappers.getMapper(ConvertUtil.class);

    OrderDTO map(OrderDO orderDO);

}

通過mvn命令編譯后,在target的classes目錄ConvertUtil統(tǒng)計(jì)目錄下會(huì)生成一個(gè)ConvertUtilImpl類澈歉,里面包括模型拷貝的邏輯展鸡。

上面map方法對應(yīng)生成的代碼如下

    public OrderDTO map(OrderDO orderDO) {
        if (orderDO == null) {
            return null;
        } else {
            OrderDTO orderDTO = new OrderDTO();
            orderDTO.setOrderCode(orderDO.getOrderCode());
            orderDTO.setAddress(orderDO.getAddress());
            return orderDTO;
        }
    }

MapStruct最基礎(chǔ)的功能講解完畢,下面介紹MapStruct一系列遍歷的功能

更多功能

名稱映射

有時(shí)候2個(gè)實(shí)體之間的名字不完全統(tǒng)一埃难,但是可能代表的是一個(gè)含義莹弊,我們可以在方法上面加上@Mapping注解來實(shí)現(xiàn)不同名稱屬性的賦值

將上面OrderDTO的address改為receiverAddress,然后在對應(yīng)map上增加@Mapping注解

    @Mapping(source = "address",target = "receiverAddress")
    OrderDTO map(OrderDO orderDO);

生成字節(jié)碼如下

    @Override
    public OrderDTO map(OrderDO orderDO) {
        if ( orderDO == null ) {
            return null;
        }

        OrderDTO orderDTO = new OrderDTO();

        orderDTO.setReceiverAddress( orderDO.getAddress() );
        orderDTO.setOrderCode( orderDO.getOrderCode() );

        return orderDTO;
    }

傳入返回

上面的基礎(chǔ)功能涡尘,轉(zhuǎn)換后的模型是通過方法結(jié)果返回的忍弛,MapStruct也支持對傳入對象的賦值

    @Mapping(source = "address",target = "receiverAddress")
    void map(OrderDO orderDO,@MappingTarget OrderDTO orderDTO);

對應(yīng)字節(jié)碼如下

    @Override
    public void map(OrderDO orderDO, OrderDTO orderDTO) {
        if ( orderDO == null ) {
            return;
        }

        orderDTO.setReceiverAddress( orderDO.getAddress() );
        orderDTO.setOrderCode( orderDO.getOrderCode() );
    }

多合一

上面可以看到,OrderDO轉(zhuǎn)換成OrderDTO的時(shí)候考抄,我沒有處理orderWareDetailDTOS细疚,MapStruct支持多個(gè)對象轉(zhuǎn)換為一個(gè)對象。

    @Mapping(source = "orderDO.address",target = "receiverAddress")
    @Mapping(source = "orderWareDetailDTOList",target = "orderWareDetailDTOS")
    OrderDTO map(OrderDO orderDO,List<OrderWareDetailDTO> orderWareDetailDTOList);

對應(yīng)字節(jié)碼如下

    @Override
    public OrderDTO map(OrderDO orderDO, List<OrderWareDetailDTO> orderWareDetailDTOList) {
        if ( orderDO == null && orderWareDetailDTOList == null ) {
            return null;
        }

        OrderDTO orderDTO = new OrderDTO();

        if ( orderDO != null ) {
            orderDTO.setReceiverAddress( orderDO.getAddress() );
            orderDTO.setOrderCode( orderDO.getOrderCode() );
        }
        if ( orderWareDetailDTOList != null ) {
            List<OrderWareDetailDTO> list = orderWareDetailDTOList;
            if ( list != null ) {
                orderDTO.setOrderWareDetailDTOS( new ArrayList<OrderWareDetailDTO>( list ) );
            }
        }

        return orderDTO;
    }

嵌套轉(zhuǎn)換

可以注意到我上面的OrderDTO有orderWareDetailDTOS這個(gè)屬性川梅,并且是一個(gè)List疯兼,然后我有下面對應(yīng)的OrderVO需要進(jìn)行轉(zhuǎn)換。

@Data
public class OrderVO {

    private String orderCode;

    private String address;

    private List<OrderWareDetailVO> orderWareDetailVOList;

}

可以看到對應(yīng)2個(gè)模型內(nèi)的list屬性類型是不一樣的贫途,其他3中轉(zhuǎn)換框架做不到這個(gè)吧彪,或者做起來挺麻煩。但是在MapStruct只要加一下2個(gè)接口方法即可丢早。

    @Mapping(target = "orderWareDetailVOList",source = "orderWareDetailDTOS")
    @Mapping(target = "address",source = "receiverAddress")
    OrderVO map(OrderDTO orderDTO);

    OrderWareDetailVO map(OrderWareDetailDTO orderWareDetailDTO);

生成的字節(jié)碼如下

    @Override
    public OrderVO map(OrderDTO orderDTO) {
        if ( orderDTO == null ) {
            return null;
        }

        OrderVO orderVO = new OrderVO();

        orderVO.setOrderWareDetailVOList( orderWareDetailDTOListToOrderWareDetailVOList( orderDTO.getOrderWareDetailDTOS() ) );
        orderVO.setAddress( orderDTO.getReceiverAddress() );
        orderVO.setOrderCode( orderDTO.getOrderCode() );

        return orderVO;
    }

    @Override
    public OrderWareDetailVO map(OrderWareDetailDTO orderWareDetailDTO) {
        if ( orderWareDetailDTO == null ) {
            return null;
        }

        OrderWareDetailVO orderWareDetailVO = new OrderWareDetailVO();

        orderWareDetailVO.setWareCode( orderWareDetailDTO.getWareCode() );

        return orderWareDetailVO;
    }

    protected List<OrderWareDetailVO> orderWareDetailDTOListToOrderWareDetailVOList(List<OrderWareDetailDTO> list) {
        if ( list == null ) {
            return null;
        }

        List<OrderWareDetailVO> list1 = new ArrayList<OrderWareDetailVO>( list.size() );
        for ( OrderWareDetailDTO orderWareDetailDTO : list ) {
            list1.add( map( orderWareDetailDTO ) );
        }

        return list1;
    }

可以看出來這個(gè)框架不是簡單的生成對應(yīng)賦值的字節(jié)碼姨裸,如果它遇到不能解析的轉(zhuǎn)換,它會(huì)從所在接口的方法中去檢查是否有對應(yīng)轉(zhuǎn)換邏輯怨酝。

其他

其他還有很多特性傀缩,比如對于數(shù)字類型,只要屬性名一樣凫碌,它會(huì)自動(dòng)轉(zhuǎn)換扑毡,當(dāng)然高精度轉(zhuǎn)低精度可能會(huì)損失精度。也包括對各種時(shí)間盛险,以及string和int之間的互轉(zhuǎn)瞄摊。我上面的舉的例子是我在開發(fā)中比較常見的一些點(diǎn),你有更加復(fù)雜的場景推薦閱讀官方文檔苦掘。

原理

別看MapStruct和lombok都是通過注解來做文章换帜,但是他們的原理是不同的。lombok是修改原有類的字節(jié)碼鹤啡,用的是jvm提供的Instrument機(jī)制惯驼。
看它的META-INF/MANIFEST.MF文件就知道了

Manifest-Version: 1.0
Ant-Version: Apache Ant 1.7.1
Created-By: 14.3-b01-101 (Apple Inc.)
Premain-Class: lombok.launch.Agent
Agent-Class: lombok.launch.Agent
Can-Redefine-Classes: true
Main-Class: lombok.launch.Main
Lombok-Version: 1.18.0

而MapStruct用到了APT(Annotation Processing Tool 簡稱,即注解處理器)這個(gè)技術(shù),通過spi提供實(shí)現(xiàn)類祟牲,會(huì)在編譯階段處理特定注解隙畜。

我們上面的配置方式?jīng)]有看不到注解解析器的依賴,maven幫我們做了這個(gè)事说贝,通過下面依賴把MapStruct APT的jar包引入項(xiàng)目

        <dependency>
            <groupId>org.mapstruct</groupId>
            <artifactId>mapstruct-processor</artifactId>
            <version>${org.mapstruct.version}</version>
            <scope>provided</scope>
        </dependency>

在jar包的services目錄下可以看到j(luò)avax.annotation.processing.Processor文件,內(nèi)容為

org.mapstruct.ap.MappingProcessor

具體怎么處理大家自己研究议惰。

寫這些框架的人,可見對java每個(gè)版本的特性有多了解乡恕。

參考

Java反射到底慢在哪
官方文檔多多了解
Java中的APT的工作過程

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末言询,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子傲宜,更是在濱河造成了極大的恐慌运杭,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,383評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件函卒,死亡現(xiàn)場離奇詭異辆憔,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)谆趾,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,522評論 3 385
  • 文/潘曉璐 我一進(jìn)店門躁愿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人沪蓬,你說我怎么就攤上這事彤钟。” “怎么了跷叉?”我有些...
    開封第一講書人閱讀 157,852評論 0 348
  • 文/不壞的土叔 我叫張陵逸雹,是天一觀的道長。 經(jīng)常有香客問我云挟,道長梆砸,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,621評論 1 284
  • 正文 為了忘掉前任园欣,我火速辦了婚禮帖世,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘沸枯。我一直安慰自己日矫,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,741評論 6 386
  • 文/花漫 我一把揭開白布绑榴。 她就那樣靜靜地躺著哪轿,像睡著了一般。 火紅的嫁衣襯著肌膚如雪翔怎。 梳的紋絲不亂的頭發(fā)上窃诉,一...
    開封第一講書人閱讀 49,929評論 1 290
  • 那天杨耙,我揣著相機(jī)與錄音,去河邊找鬼飘痛。 笑死珊膜,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的敦冬。 我是一名探鬼主播辅搬,決...
    沈念sama閱讀 39,076評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼唯沮,長吁一口氣:“原來是場噩夢啊……” “哼脖旱!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起介蛉,我...
    開封第一講書人閱讀 37,803評論 0 268
  • 序言:老撾萬榮一對情侶失蹤萌庆,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后币旧,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體践险,經(jīng)...
    沈念sama閱讀 44,265評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,582評論 2 327
  • 正文 我和宋清朗相戀三年吹菱,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了巍虫。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,716評論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡鳍刷,死狀恐怖占遥,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情输瓜,我是刑警寧澤瓦胎,帶...
    沈念sama閱讀 34,395評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站尤揣,受9級特大地震影響搔啊,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜北戏,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,039評論 3 316
  • 文/蒙蒙 一负芋、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧嗜愈,春花似錦旧蛾、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,798評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至拌阴,卻和暖如春绍绘,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,027評論 1 266
  • 我被黑心中介騙來泰國打工陪拘, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留厂镇,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,488評論 2 361
  • 正文 我出身青樓左刽,卻偏偏與公主長得像捺信,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子欠痴,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,612評論 2 350