成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

MapStruct教程-枚舉的五種用法

開發(fā) 前端
日常開發(fā)中,我們經(jīng)常會用到枚舉,有時(shí)候會涉及枚舉之間的映射、枚舉與int或String之間的映射等。本文一起看下,MapStruct中如何實(shí)現(xiàn)。

你好,我是看山。

日常開發(fā)中,我們經(jīng)常會用到枚舉,有時(shí)候會涉及枚舉之間的映射、枚舉與int或String之間的映射等。本文一起看下,MapStruct中如何實(shí)現(xiàn)。

一、將一個(gè)枚舉映射到另一個(gè)枚舉

(一)用例說明

  • 在 REST API 中,將外部API狀態(tài)碼轉(zhuǎn)換為我們應(yīng)用內(nèi)部的狀態(tài)枚舉;
  • 與第三方庫集成時(shí),兩個(gè)服務(wù)間枚舉定義不同,通常需要處理枚舉映射。

(二)使用MapStruct實(shí)現(xiàn)映射

這里我們會用到@ValueMapping注解,可以實(shí)現(xiàn)源常量值到目標(biāo)常量值的映射。

我們看下實(shí)際應(yīng)用。首先定義一個(gè)表示交通信號的枚舉TrafficSignal:

public enum TrafficSignal {
    OFF,
    STOP,
    GO;
}

在定義一個(gè)表示道路標(biāo)記的源枚舉RoadSign:

public enum RoadSign {
    OFF,
    HALT,
    MOVE;
}

接下來,我們定義一個(gè)映射:

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

    @ValueMapping(target = "GO", source = "MOVE")
    @ValueMapping(target = "STOP", source = "HALT")
    TrafficSignal toTrafficSignal(RoadSign source);
}

看下生成的實(shí)現(xiàn):

public class TrafficSignalMapperImpl implements TrafficSignalMapper {

    @Override
    public TrafficSignal toTrafficSignal(RoadSign source) {
        if ( source == null ) {
            return null;
        }

        TrafficSignal trafficSignal;

        switch ( source ) {
            case MOVE: trafficSignal = TrafficSignal.GO;
            break;
            case HALT: trafficSignal = TrafficSignal.STOP;
            break;
            case OFF: trafficSignal = TrafficSignal.OFF;
            break;
            default: throw new IllegalArgumentException( "Unexpected enum constant: " + source );
        }

        return trafficSignal;
    }
}

可以看到,因?yàn)镺FF是名字相同,雖然沒有定義映射關(guān)系,MapStruct會自動匹配。剩下兩個(gè)枚舉值根據(jù)我們的定義匹配上了。

這里需要注意的是,我們需要確保將源枚舉的所有值都映射到目標(biāo)枚舉,如果沒有完全匹配上,會走到default分支,拋出IllegalArgumentException異常。

二、將字符串映射到枚舉

我們繼續(xù)看下字符串與枚舉之間的映射。有了前面的基礎(chǔ),我們這里直接上手,還是使用@ValueMapping注解,字符串的值都是小寫,需要轉(zhuǎn)換為TrafficSignal枚舉:

@ValueMapping(target = "OFF", source = "off")
@ValueMapping(target = "GO", source = "move")
@ValueMapping(target = "STOP", source = "halt")
TrafficSignal stringToTrafficSignal(String source);

我們看下生成的代碼:

@Override
public TrafficSignal stringToTrafficSignal(String source) {
    if ( source == null ) {
        return null;
    }

    TrafficSignal trafficSignal;

    switch ( source ) {
        case "off": trafficSignal = TrafficSignal.OFF;
        break;
        case "move": trafficSignal = TrafficSignal.GO;
        break;
        case "halt": trafficSignal = TrafficSignal.STOP;
        break;
        case "OFF": trafficSignal = TrafficSignal.OFF;
        break;
        case "STOP": trafficSignal = TrafficSignal.STOP;
        break;
        case "GO": trafficSignal = TrafficSignal.GO;
        break;
        default: throw new IllegalArgumentException( "Unexpected enum constant: " + source );
    }

    return trafficSignal;
}

可以看到,除了我們定義的三個(gè)映射,MapStruct還會自動將枚舉的name()也作為映射依據(jù),換句話說,如果我們輸入的字符串與枚舉正好是一一對應(yīng)的,那就可以不用定義映射關(guān)系了。

三、處理自定義名稱轉(zhuǎn)換

還有一種情況是,需要映射的枚舉值有統(tǒng)一的約束,比如遵循不同的大小寫、前綴或后綴等,比如,一個(gè)信號可以是Go、go、GO、Go_Value、Value_Go等。

(一)后綴

假如我們的目標(biāo)枚舉相較于源枚舉有統(tǒng)一的后綴,比如:GO到GO_VALUE。

public enum TrafficSignalSuffixed {
    OFF_VALUE,
    STOP_VALUE,
    GO_VALUE
}

此時(shí),我們可以用到@EnumMapping注解,定義名稱轉(zhuǎn)換策略是后綴,然后定義后綴值:

@EnumMapping(nameTransformationStrategy = SUFFIX_TRANSFORMATION, configuration = "_VALUE")
TrafficSignalSuffixed applySuffix(TrafficSignal source);

@EnumMapping為枚舉類型定義自定義映射,nameTransformationStrategy指定在映射之前應(yīng)用于枚舉常量名稱的轉(zhuǎn)換策略,并使用configuration定義控制值。

生成結(jié)果是:

@Override
public TrafficSignalSuffixed applySuffix(TrafficSignal source) {
    if ( source == null ) {
        return null;
    }

    TrafficSignalSuffixed trafficSignalSuffixed;

    switch ( source ) {
        case OFF: trafficSignalSuffixed = TrafficSignalSuffixed.OFF_VALUE;
        break;
        case STOP: trafficSignalSuffixed = TrafficSignalSuffixed.STOP_VALUE;
        break;
        case GO: trafficSignalSuffixed = TrafficSignalSuffixed.GO_VALUE;
        break;
        default: throw new IllegalArgumentException( "Unexpected enum constant: " + source );
    }

    return trafficSignalSuffixed;
}

需要注意,@EnumMapping應(yīng)用的場景是枚舉值完全符合指定策略,如果其中有某個(gè)值不符合,編譯時(shí)會出現(xiàn)異常“The following constants from the source enum have no corresponding constant in the target enum and must be be mapped via adding additional mappings: xxx.”

(二)前綴

假如我們的目標(biāo)枚舉相較于源枚舉有統(tǒng)一的前綴綴,比如:GO到VALUE_GO。

public enum TrafficSignalPrefixed {
    VALUE_OFF,
    VALUE_STOP,
    VALUE_GO;
}

定義映射:

@EnumMapping(nameTransformationStrategy = PREFIX_TRANSFORMATION, configuration = "VALUE_")
TrafficSignalPrefixed applyPrefix(TrafficSignal source);

PREFIX_TRANSFORMATION是告訴MapStruct,需要在源枚舉增加前綴VALUE_。

生成的代碼是:

public TrafficSignalPrefixed applyPrefix(TrafficSignal source) {
    if ( source == null ) {
        return null;
    }

    TrafficSignalPrefixed trafficSignalPrefixed;

    switch ( source ) {
        case OFF: trafficSignalPrefixed = TrafficSignalPrefixed.VALUE_OFF;
        break;
        case STOP: trafficSignalPrefixed = TrafficSignalPrefixed.VALUE_STOP;
        break;
        case GO: trafficSignalPrefixed = TrafficSignalPrefixed.VALUE_GO;
        break;
        default: throw new IllegalArgumentException( "Unexpected enum constant: " + source );
    }

    return trafficSignalPrefixed;
}

(三)去除后綴

假如我們的目標(biāo)枚舉相較于源枚舉缺少統(tǒng)一的后綴,比如:GO_VALUE到GO。

我們可以使用STRIP_SUFFIX_TRANSFORMATION指定去除后綴:

@EnumMapping(nameTransformationStrategy = STRIP_SUFFIX_TRANSFORMATION, configuration = "_VALUE")
TrafficSignal stripSuffix(TrafficSignalSuffixed source);

(四)去除前綴

假如我們的目標(biāo)枚舉相較于源枚舉缺少統(tǒng)一的前綴,比如:VALUE_GO到GO。

我們可以使用STRIP_PREFIX_TRANSFORMATION指定去除前綴:

@EnumMapping(nameTransformationStrategy = STRIP_PREFIX_TRANSFORMATION, configuration = "VALUE_")
TrafficSignal stripPrefix(TrafficSignalPrefixed source);

(五)小寫

假如我們的目標(biāo)枚舉是源枚舉的小寫,比如:GO變?yōu)間o:

public enum TrafficSignalLowercase {
    off,
    stop,
    go;
}

我們需要使用CASE_TRANSFORMATION策略,并定義策略是lower。

定義映射:

@EnumMapping(nameTransformationStrategy = CASE_TRANSFORMATION, configuration = "lower")
TrafficSignalLowercase applyLowercase(TrafficSignal source);

(六)大寫

假如我們的目標(biāo)枚舉是源枚舉的小寫,比如:go變?yōu)镚O:

還是使用CASE_TRANSFORMATION策略,并定義策略是upper。

@EnumMapping(nameTransformationStrategy = CASE_TRANSFORMATION, configuration = "upper")
TrafficSignal applyUppercase(TrafficSignalLowercase source);

(七)首字母大寫

我們還可以指定首字母大寫的映射,例如,go變?yōu)镚o。

定義下目標(biāo)枚舉

public enum TrafficSignalCapital {
    Off,
    Stop,
    Go;
}

還是使用CASE_TRANSFORMATION策略,并定義策略是capital。

@EnumMapping(nameTransformationStrategy = CASE_TRANSFORMATION, configuration = "capital")
TrafficSignalCapital lowercaseToCapital(TrafficSignalLowercase source);

四、枚舉映射的其他用例

還有些場景中,我們需要將枚舉映射到其他類型,接下來,一起看看如何處理。

(一)將枚舉映射到字符串

定義映射:

@ValueMapping(target = "off", source = "OFF")
@ValueMapping(target = "go", source = "GO")
@ValueMapping(target = "stop", source = "STOP")
String trafficSignalToString(TrafficSignal source);

我們使用@ValueMapping將枚舉值映射到字符串,其實(shí)是和從字符串轉(zhuǎn)枚舉相似的配置邏輯。

(二)將枚舉映射到整數(shù)或其他數(shù)字類型

因?yàn)閿?shù)字類型存在多個(gè)構(gòu)造函數(shù),直接映射到整數(shù)可能會導(dǎo)致歧義。可以定義一個(gè)具有整數(shù)屬性的類來解決這個(gè)問題。

定義一個(gè)包裝類:

public class TrafficSignalNumber {
    private Integer number;
}

使用默認(rèn)方法將枚舉映射到整數(shù):

@Mapping(target = "number", source = ".")
TrafficSignalNumber trafficSignalToTrafficSignalNumber(TrafficSignal source);

生成的代碼是:

public TrafficSignalNumber trafficSignalToTrafficSignalNumber(TrafficSignal source) {
    if ( source == null ) {
        return null;
    }

    TrafficSignalNumber trafficSignalNumber = new TrafficSignalNumber();

    if ( source != null ) {
        trafficSignalNumber.setNumber( source.ordinal() );
    }

    return trafficSignalNumber;
}

五、處理未知枚舉值

前面提到過,在處理枚舉值值,當(dāng)有未映射的枚舉值時(shí),MapStruct會拋出異常。

不過,很多時(shí)候,當(dāng)映射失敗的時(shí)候,我們需要有不同的操作,比如:設(shè)置默認(rèn)值、設(shè)置空值、拋出異常等。

(一)未映射拋出異常

拋出異常是默認(rèn)行為,前面的示例中都是屬于這種類型。

(二)映射剩余屬性

比如,我們有一個(gè)簡單的交通信號枚舉:

public enum SimpleTrafficSignal {
    OFF,
    ON;
}

需要將toSimpleTrafficSignal映射到SimpleTrafficSignal,但是MapStruct要求所有枚舉值都需要映射,不能遺漏,所以我們可以這樣寫:

@ValueMapping(target = "OFF", source = "OFF")
@ValueMapping(target = "OFF", source = "STOP")
@ValueMapping(target = "ON", source = "GO")
SimpleTrafficSignal toSimpleTrafficSignal(TrafficSignal source);

我們顯式地將STOP和OFF都映射到OFF,但是如果值特別多的時(shí)候,這樣寫就顯得很傻,我們可以使用ANY_REMAINING配置:

@ValueMapping(target = "ON", source = "GO")
@ValueMapping(target = "OFF", source = ANY_REMAINING)
SimpleTrafficSignal toSimpleTrafficSignalWithRemaining(TrafficSignal source);

生成的代碼是:

public SimpleTrafficSignal toSimpleTrafficSignalWithRemaining(TrafficSignal source) {
    if ( source == null ) {
        return null;
    }

    SimpleTrafficSignal simpleTrafficSignal;

    switch ( source ) {
        case GO: simpleTrafficSignal = SimpleTrafficSignal.ON;
        break;
        case OFF: simpleTrafficSignal = SimpleTrafficSignal.OFF;
        break;
        default: simpleTrafficSignal = SimpleTrafficSignal.OFF;
    }

    return simpleTrafficSignal;
}

也就是,除了GO明確映射外,其他的都映射為OFF。

(三)映射未映射的屬性

我們可以所有未映射到值全部映射為指定的枚舉,比如,所有沒有配置的都映射為OFF,我們可以使用ANY_UNMAPPED配置:

@ValueMapping(target = "ON", source = "GO")
@ValueMapping(target = "OFF", source = ANY_UNMAPPED)
SimpleTrafficSignal toSimpleTrafficSignalWithUnmapped(TrafficSignal source);

生成的代碼是:

@Override
public SimpleTrafficSignal toSimpleTrafficSignalWithUnmapped(TrafficSignal source) {
    if ( source == null ) {
        return null;
    }

    SimpleTrafficSignal simpleTrafficSignal;

    switch ( source ) {
        case GO: simpleTrafficSignal = SimpleTrafficSignal.ON;
        break;
        default: simpleTrafficSignal = SimpleTrafficSignal.OFF;
    }

    return simpleTrafficSignal;
}

(四)處理空值

MapStruct可以使用NULL關(guān)鍵字處理空的源和空的目標(biāo)。

假設(shè)我們需要將空輸入映射到OFF,將GO映射到ON,將任何其他未映射的值映射到空。

我們可以這樣定義映射:

@ValueMapping(target = "OFF", source = NULL)
@ValueMapping(target = "ON", source = "GO")
@ValueMapping(target = NULL, source = MappingConstants.ANY_UNMAPPED)
SimpleTrafficSignal toSimpleTrafficSignalWithNullHandling(TrafficSignal source);

生成代碼是:

public SimpleTrafficSignal toSimpleTrafficSignalWithNullHandling(TrafficSignal source) {
    if ( source == null ) {
        return SimpleTrafficSignal.OFF;
    }

    SimpleTrafficSignal simpleTrafficSignal;

    switch ( source ) {
        case GO: simpleTrafficSignal = SimpleTrafficSignal.ON;
        break;
        default: simpleTrafficSignal = null;
    }

    return simpleTrafficSignal;
}

(五)指定值拋出異常

還有一種場景,就是為空或者未映射時(shí),拋出異常,我們可以使用THROW_EXCEPTION策略:

定義映射:

@ValueMapping(target = "ON", source = "GO")
@ValueMapping(target = MappingConstants.THROW_EXCEPTION, source = MappingConstants.ANY_UNMAPPED)
@ValueMapping(target = MappingConstants.THROW_EXCEPTION, source = MappingConstants.NULL)
SimpleTrafficSignal toSimpleTrafficSignalWithExceptionHandling(TrafficSignal source);

生成的代碼是:

@Override
public SimpleTrafficSignal toSimpleTrafficSignalWithExceptionHandling(TrafficSignal source) {
    if ( source == null ) {
        throw new IllegalArgumentException( "Unexpected enum constant: " + source );
    }

    SimpleTrafficSignal simpleTrafficSignal;

    switch ( source ) {
        case GO: simpleTrafficSignal = SimpleTrafficSignal.ON;
        break;
        default: throw new IllegalArgumentException( "Unexpected enum constant: " + source );
    }

    return simpleTrafficSignal;
}

責(zé)任編輯:武曉燕 來源: 看山的小屋
相關(guān)推薦

2025-01-16 00:00:00

MapStruct映射

2023-07-07 08:17:48

2011-10-08 15:49:52

Java

2025-01-13 00:00:00

MapStruct繼承關(guān)系Java

2013-01-07 10:14:06

JavaJava枚舉

2011-10-10 17:15:52

Java

2021-09-22 12:56:19

編程技能Golang

2024-05-20 12:00:00

Python列表推導(dǎo)式

2025-01-10 00:00:00

MapStructArticlePerson

2025-01-24 00:00:00

MapStruct子類型Mapper

2017-07-24 10:15:32

2023-05-22 08:03:28

JavaScrip枚舉定義

2025-05-26 04:00:00

2020-09-10 07:00:00

人工智能AI機(jī)器學(xué)習(xí)

2010-12-14 09:27:44

綠色網(wǎng)絡(luò)

2020-09-25 08:00:57

Kubernetes

2011-10-24 09:25:54

蘋果Siri調(diào)戲

2022-05-10 08:08:01

find命令Linux

2009-06-19 18:26:38

Spring事務(wù)配置

2010-07-19 16:00:51

職場
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 久久成人免费 | 一区二区视频免费观看 | 欧美日本韩国一区二区 | 欧美日韩一区二区在线 | 日韩在线小视频 | 最新av中文字幕 | 日本特黄a级高清免费大片 成年人黄色小视频 | 久久一级| 日韩精品一区二 | 亚洲一区二区日韩 | 在线国产一区 | 欧美一级片免费看 | av在线二区| 在线小视频 | 亚洲高清视频在线观看 | 日韩综合网 | 一本一道久久a久久精品蜜桃 | 中文字幕 国产 | 国产精品久久久久久久一区探花 | 欧美8一10sex性hd| 久久精品亚洲精品 | 在线观看国产视频 | 国产在线一区二区三区 | 精品久久久久久久久久久久久久 | 欧美性生活免费 | 精品av天堂毛片久久久借种 | 91精品久久久久久久久 | 亚洲性视频在线 | 国产精品免费看 | 日韩中文在线观看 | 一级黄色毛片免费 | 精品96久久久久久中文字幕无 | 中文字幕一区在线观看视频 | 国产精品国产精品国产专区不片 | 久久亚洲一区 | 久久综合一区 | 亚洲男人网 | 亚洲人成人一区二区在线观看 | 久久精品久久综合 | 国产精品色一区二区三区 | 亚洲精品乱码 |