欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

SpringBoot在RequestBody中使用枚舉參數(shù)案例詳解

 更新時間:2021年09月03日 11:31:00   作者:沉潛飛動  
這篇文章主要介紹了SpringBoot在RequestBody中使用枚舉參數(shù)案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下

前文說到 優(yōu)雅的使用枚舉參數(shù)實現(xiàn)原理,本文繼續(xù)說一下如何在 RequestBody 中優(yōu)雅使用枚舉。

本文先上實戰(zhàn),說一下如何實現(xiàn)。在 優(yōu)雅的使用枚舉參數(shù) 代碼的基礎(chǔ)上,我們繼續(xù)實現(xiàn)。

確認(rèn)需求

需求與前文類似,只不過這里需要是在 RequestBody 中使用。與前文不同的是,這種請求是通過 Http Body 的方式傳輸?shù)胶蠖?,通常?json 或 xml 格式,Spring 默認(rèn)借助 Jackson 反序列化為對象。

同樣的,我們需要在枚舉中定義 int 類型的 id、String 類型的 code,id 取值不限于序號(即從 0 開始的 orinal 數(shù)據(jù)),code 不限于 name。客戶端請求過程中,可以傳 id,可以傳 code,也可以傳 name。服務(wù)端只需要在對象中定義一個枚舉參數(shù),不需要額外的轉(zhuǎn)換,即可得到枚舉值。

好了,接下來我們定義一下枚舉對象。

定義枚舉和對象

先定義我們的枚舉類GenderIdCodeEnum,包含 id 和 code 兩個屬性:

public enum GenderIdCodeEnum implements IdCodeBaseEnum {
    MALE(1, "male"),
    FEMALE(2, "female");

    private final Integer id;
    private final String code;

    GenderIdCodeEnum(Integer id, String code) {
        this.id = id;
        this.code = code;
    }

    @Override
    public String getCode() {
        return code;
    }

    @Override
    public Integer getId() {
        return id;
    }
}

這個枚舉類的要求與前文一致,不清楚的可以再去看一下。

在定義一個包裝類GenderIdCodeRequestBody,用于接收 json 數(shù)據(jù)的請求體:

@Data
public class GenderIdCodeRequestBody {
    private String name;
    private GenderIdCodeEnum gender;
    private long timestamp;
}

除了GenderIdCodeEnum參數(shù)外,其他都是示例,所以隨便定義一下。

實現(xiàn)轉(zhuǎn)換邏輯

前奏鋪墊好,接下來入正題了。Jackson 提供了兩種方案:

  • 方案一:精準(zhǔn)攻擊,指定需要轉(zhuǎn)換的字段,不影響其他類對象中的字段
  • 方案二:全范圍攻擊,所有借助 Jackson 反序列化的枚舉字段,全部具備自動轉(zhuǎn)換功能

方案一:精準(zhǔn)攻擊

這種方案中,我們首先需要實現(xiàn)JsonDeserialize抽象類:

public class IdCodeToEnumDeserializer extends JsonDeserializer<BaseEnum> {
    @Override
    public BaseEnum deserialize(JsonParser jsonParser, DeserializationContext deserializationContext)
            throws IOException {
        final String param = jsonParser.getText();// 1
        final JsonStreamContext parsingContext = jsonParser.getParsingContext();// 2
        final String currentName = parsingContext.getCurrentName();// 3
        final Object currentValue = parsingContext.getCurrentValue();// 4
        try {
            final Field declaredField = currentValue.getClass().getDeclaredField(currentName);// 5
            final Class<?> targetType = declaredField.getType();// 6
            final Method createMethod = targetType.getDeclaredMethod("create", Object.class);// 7
            return (BaseEnum) createMethod.invoke(null, param);// 8
        } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException | NoSuchFieldException e) {
            throw new CodeBaseException(ErrorResponseEnum.PARAMS_ENUM_NOT_MATCH, new Object[] {param}, "", e);
        }
    }
}

然后在指定枚舉字段上定義@JsonDeserialize注解,比如:

@JsonDeserialize(using = IdCodeToEnumDeserializer.class)
private GenderIdCodeEnum gender;

具體說一下每行的作用:

  1. 獲取參數(shù)值。根據(jù)需要,此處可能是 id、code 或 name,也就是源值,需要將其轉(zhuǎn)換為枚舉;
  2. 獲取轉(zhuǎn)換上線文,這個是為 3、4 步做準(zhǔn)備的;
  3. 獲取標(biāo)記@JsonDeserialize注解的字段,此時currentName的值是gender;
  4. 獲取包裝對象,也就是GenderIdCodeRequestBody對象;
  5. 根據(jù)包裝對象的Class對象,以及字段名gender獲取Field對象,為第 5 步做準(zhǔn)備;
  6. 獲取gender字段對應(yīng)的枚舉類型,也即是GenderIdCodeEnum。之所以這樣做,是要實現(xiàn)一個通用的反序列化類;
  7. 這里是寫死的一種實現(xiàn),就是在枚舉類中,需要定義一個靜態(tài)方法,方法名是create,請求參數(shù)是Object;
  8. 通過反射調(diào)用create方法,將第一步獲取的請求參數(shù)傳入。

我們來看一下枚舉類中定義的create方法:

public static GenderIdCodeEnum create(Object code) {
    final String stringCode = code.toString();
    final Integer intCode = BaseEnum.adapter(stringCode);
    for (GenderIdCodeEnum item : values()) {
        if (Objects.equals(stringCode, item.name())) {
            return item;
        }
        if (Objects.equals(item.getCode(), stringCode)) {
            return item;
        }
        if (Objects.equals(item.getId(), intCode)) {
            return item;
        }
    }
    return null;
}

為了性能考慮,我們可以提前定義三組 map,分別以 id、code、name 為 key,以枚舉值為 value,這樣就可以通過 O(1) 的時間復(fù)雜度返回了??梢詤⒖记拔牡?code>Converter類的實現(xiàn)邏輯。

這樣,我們就可以實現(xiàn)精準(zhǔn)轉(zhuǎn)換了。

方案二:全范圍攻擊

這種方案是全范圍攻擊了,只要是 Jackson 參與的反序列化,只要其中有目標(biāo)枚舉參數(shù),就會受到這種進(jìn)入這種方案的邏輯中。這種方案是在枚舉類中定義一個靜態(tài)轉(zhuǎn)換方法,通過@JsonCreator注解注釋,Jackson 就會自動轉(zhuǎn)換了。

這個方法的定義與方案一中的create方法完全一致,所以只需要在create方法上加上注解即可:

@JsonCreator(mode = Mode.DELEGATING)
public static GenderIdCodeEnum create(Object code) {
    final String stringCode = code.toString();
    final Integer intCode = BaseEnum.adapter(stringCode);
    for (GenderIdCodeEnum item : values()) {
        if (Objects.equals(stringCode, item.name())) {
            return item;
        }
        if (Objects.equals(item.getCode(), stringCode)) {
            return item;
        }
        if (Objects.equals(item.getId(), intCode)) {
            return item;
        }
    }
    return null;
}

其中Mode類有四個值:DEFAULT、DELEGATING、PROPERTIESDISABLED,這四種的差別會在原理篇中說明。還是那句話,對于應(yīng)用類技術(shù),我們可以先知其然,再知其所以然,也一定要知其所以然。

測試

先定義一個 controller 方法:

@PostMapping("gender-id-code-request-body")
public GenderIdCodeRequestBody bodyGenderIdCode(@RequestBody GenderIdCodeRequestBody genderRequest) {
    genderRequest.setTimestamp(System.currentTimeMillis());
    return genderRequest;
}

然后定義測試用例,還是借助 JUnit5:

@ParameterizedTest
@ValueSource(strings = {"\"MALE\"", "\"male\"", "\"1\"", "1"})
void postGenderIdCode(String gender) throws Exception {
    final String result = mockMvc.perform(
            MockMvcRequestBuilders.post("/echo/gender-id-code-request-body")
                    .contentType(MediaType.APPLICATION_JSON_UTF8)
                    .accept(MediaType.APPLICATION_JSON_UTF8)
                    .content("{\"gender\": " + gender + ", \"name\": \"看山\"}")
    )
            .andExpect(MockMvcResultMatchers.status().isOk())
            .andDo(MockMvcResultHandlers.print())
            .andReturn()
            .getResponse()
            .getContentAsString();

    ObjectMapper objectMapper = new ObjectMapper();
    final GenderIdCodeRequestBody genderRequest = objectMapper.readValue(result, GenderIdCodeRequestBody.class);
    Assertions.assertEquals(GenderIdCodeEnum.MALE, genderRequest.getGender());
    Assertions.assertEquals("看山", genderRequest.getName());
    Assertions.assertTrue(genderRequest.getTimestamp() > 0);
}

文末總結(jié)

本文主要說明了如何在 RequestBody 中優(yōu)雅的使用枚舉參數(shù),借助了 Jackson 的反序列化擴(kuò)展,可以定制類型轉(zhuǎn)換邏輯。礙于文章篇幅,沒有羅列大段代碼。關(guān)注公號「看山的小屋」回復(fù) spring 可以獲取源碼。關(guān)注我,下一篇我們進(jìn)入原理篇。

推薦閱讀

到此這篇關(guān)于SpringBoot在RequestBody中使用枚舉參數(shù)案例詳解的文章就介紹到這了,更多相關(guān)SpringBoot在RequestBody中使用枚舉參數(shù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot自定義注解開發(fā)指南

    SpringBoot自定義注解開發(fā)指南

    在開發(fā)SpringBoot程序的過程中,有可能與其他業(yè)務(wù)系統(tǒng)進(jìn)行對接開發(fā),獲取封裝公共的API接口等等,下面這篇文章主要給大家介紹了關(guān)于SpringBoot自定義注解的相關(guān)資料,需要的朋友可以參考下
    2022-06-06
  • Java中Map集合(接口)的基本方法程序演示

    Java中Map集合(接口)的基本方法程序演示

    這篇文章主要為大家詳細(xì)介紹了Java中Map集合的基本方法程序演示,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-08-08
  • Java IO流常用字節(jié)字符流原理解析

    Java IO流常用字節(jié)字符流原理解析

    這篇文章主要介紹了Java IO流常用字節(jié)字符流原理解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-03-03
  • Java使用注解實現(xiàn)防止重復(fù)提交實例

    Java使用注解實現(xiàn)防止重復(fù)提交實例

    這篇文章主要介紹了Java使用注解實現(xiàn)防止重復(fù)提交實例,在一些項目中由于用戶誤操作,多次點擊表單提交按鈕,會產(chǎn)生很多次的數(shù)據(jù)交互,為了解決這一問題,本文使用注解來實現(xiàn)防止重復(fù)提交,需要的朋友可以參考下
    2023-07-07
  • Java如何實現(xiàn)圖片的疊加與拼接操作

    Java如何實現(xiàn)圖片的疊加與拼接操作

    這篇文章主要介紹了Java如何實現(xiàn)圖片的疊加與拼接操作,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-11-11
  • struts2靜態(tài)資源映射代碼示例

    struts2靜態(tài)資源映射代碼示例

    這篇文章主要介紹了struts2靜態(tài)資源映射的相關(guān)內(nèi)容,涉及了具體代碼示例,具有一定參考價值,需要的朋友可以了解下。
    2017-09-09
  • Spring深入了解常用配置應(yīng)用

    Spring深入了解常用配置應(yīng)用

    這篇文章主要給大家介紹了關(guān)于Spring的常用配置,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用springboot具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2022-07-07
  • Springboot項目使用Slf4j將日志保存到本地目錄的實現(xiàn)代碼

    Springboot項目使用Slf4j將日志保存到本地目錄的實現(xiàn)代碼

    這篇文章主要介紹了Springboot項目使用Slf4j將日志保存到本地目錄的實現(xiàn)方法,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-05-05
  • JAVA集合框架專題

    JAVA集合框架專題

    這篇文章主要介紹了JAVA集合框架的相關(guān)知識,文中講解非常細(xì)致,幫助大家更好的理解學(xué)習(xí)JAVA框架,感興趣的朋友快來了解下
    2020-06-06
  • 通過spring注解開發(fā),簡單測試單例和多例區(qū)別

    通過spring注解開發(fā),簡單測試單例和多例區(qū)別

    這篇文章主要介紹了通過spring注解開發(fā),簡單測試單例和多例區(qū)別,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-08-08

最新評論