一篇文章帶了解如何用SpringBoot在RequestBody中優(yōu)雅的使用枚舉參數(shù)
確認(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ù)端只需要在對象中定義一個(gè)枚舉參數(shù),不需要額外的轉(zhuǎn)換,即可得到枚舉值。
好了,接下來我們定義一下枚舉對象。
定義枚舉和對象
先定義我們的枚舉類GenderIdCodeEnum,包含 id 和 code 兩個(gè)屬性:
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;
}
}
這個(gè)枚舉類的要求與前文一致,不清楚的可以再去看一下。
在定義一個(gè)包裝類GenderIdCodeRequestBody,用于接收 json 數(shù)據(jù)的請求體:
@Data
public class GenderIdCodeRequestBody {
private String name;
private GenderIdCodeEnum gender;
private long timestamp;
}
除了GenderIdCodeEnum參數(shù)外,其他都是示例,所以隨便定義一下。
實(shí)現(xiàn)轉(zhuǎn)換邏輯
前奏鋪墊好,接下來入正題了。Jackson 提供了兩種方案:
- 方案一:精準(zhǔn)攻擊,指定需要轉(zhuǎn)換的字段,不影響其他類對象中的字段
- 方案二:全范圍攻擊,所有借助 Jackson 反序列化的枚舉字段,全部具備自動轉(zhuǎn)換功能
方案一:精準(zhǔn)攻擊
這種方案中,我們首先需要實(shí)現(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;
具體說一下每行的作用:
- 獲取參數(shù)值。根據(jù)需要,此處可能是 id、code 或 name,也就是源值,需要將其轉(zhuǎn)換為枚舉;
- 獲取轉(zhuǎn)換上線文,這個(gè)是為 3、4 步做準(zhǔn)備的;
- 獲取標(biāo)記@JsonDeserialize注解的字段,此時(shí)currentName的值是gender;
- 獲取包裝對象,也就是GenderIdCodeRequestBody對象;
- 根據(jù)包裝對象的Class對象,以及字段名gender獲取Field對象,為第 5 步做準(zhǔn)備;
- 獲取gender字段對應(yīng)的枚舉類型,也即是GenderIdCodeEnum。之所以這樣做,是要實(shí)現(xiàn)一個(gè)通用的反序列化類;
- 這里是寫死的一種實(shí)現(xiàn),就是在枚舉類中,需要定義一個(gè)靜態(tài)方法,方法名是create,請求參數(shù)是Object;
- 通過反射調(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) 的時(shí)間復(fù)雜度返回了。可以參考前文的Converter類的實(shí)現(xiàn)邏輯。
這樣,我們就可以實(shí)現(xiàn)精準(zhǔn)轉(zhuǎn)換了。
方案二:全范圍攻擊
這種方案是全范圍攻擊了,只要是 Jackson 參與的反序列化,只要其中有目標(biāo)枚舉參數(shù),就會受到這種進(jìn)入這種方案的邏輯中。這種方案是在枚舉類中定義一個(gè)靜態(tài)轉(zhuǎn)換方法,通過@JsonCreator注解注釋,Jackson 就會自動轉(zhuǎn)換了。
這個(gè)方法的定義與方案一中的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類有四個(gè)值:DEFAULT、DELEGATING、PROPERTIES、DISABLED,這四種的差別會在原理篇中說明。還是那句話,對于應(yīng)用類技術(shù),我們可以先知其然,再知其所以然,也一定要知其所以然。
測試
先定義一個(gè) 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)注腳本之家的更多內(nèi)容!
相關(guān)文章
mybatis插入數(shù)據(jù)后如何返回新增數(shù)據(jù)的id值
當(dāng)往mysql數(shù)據(jù)庫插入一條數(shù)據(jù)時(shí),有時(shí)候需要知道剛插入的信息,下面這篇文章主要給大家介紹了關(guān)于mybatis插入數(shù)據(jù)后如何返回新增數(shù)據(jù)id值的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06
Java?C++題解leetcode1441用棧操作構(gòu)建數(shù)組示例
這篇文章主要為大家介紹了Java?C++題解leetcode1441用棧操作構(gòu)建數(shù)組示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10
Java之NoClassDefFoundError的原因及分析
在Java開發(fā)中,經(jīng)常會遇到ClassNotFoundException和NoClassDefFoundError異常,ClassNotFoundException發(fā)生在編譯時(shí)JVM無法找到類,而NoClassDefFoundError則發(fā)生在運(yùn)行時(shí)JVM無法加載類,這兩個(gè)異常雖然原因相似,但有本質(zhì)區(qū)別2024-09-09
關(guān)于@ConfigurationProperties注解全解析
通過@ConfigurationProperties和@ConditionalOnProperty注解,可以實(shí)現(xiàn)基于配置的條件加載Bean,以此優(yōu)化Spring Boot應(yīng)用的啟動速度,在application.yml中設(shè)置配置項(xiàng),如是否加載特定的Bean(以swagger配置為例)2024-11-11
Java定時(shí)器通信協(xié)議管理模塊Timer詳解
這篇文章主要介紹了Java定時(shí)器通信協(xié)議管理模塊Timer,?Timer一般指定時(shí)器(通信協(xié)議管理模塊)人類最早使用的定時(shí)工具是沙漏或水漏,但在鐘表誕生發(fā)展成熟之后,人們開始嘗試使用這種全新的計(jì)時(shí)工具來改進(jìn)定時(shí)器,達(dá)到準(zhǔn)確控制時(shí)間的目的2022-08-08
idea運(yùn)行vue項(xiàng)目設(shè)置自定義瀏覽器方式
這篇文章主要介紹了idea運(yùn)行vue項(xiàng)目設(shè)置自定義瀏覽器方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08

