“This is the 27th day of my participation in the Gwen Challenge in November. See details of the event: The Last Gwen Challenge in 2021”
This article was included in the column “Spring Boot Actual Combat”.
Hello, I’m looking at the mountains.
After talking about using enumerations gracefully and how to implement them, this article continues with how to use enumerations gracefully in the RequestBody.
This article first on the actual combat, say how to achieve. We continue our implementation with the elegant use of enumeration argument code. If you want to get the source code, you can follow the public number “Mountain hut”, reply to Spring.
Confirm the demand
The requirements are similar to those above, except that they need to be used in the RequestBody. Unlike the previous section, this request is delivered to the back end as an Http Body, usually in JSON or XML format, and Spring deserializes objects with Jackson by default.
Similarly, we need to define an id of type int in the enumeration, and a code of type String. The value of ID is not limited to ordinals (i.e., orinal data starting from 0), and the value of code is not limited to name. During the client request, the ID, code, and name can be passed. The server only needs to define an enumeration parameter in the object, and no additional conversion is required to get the enumeration value.
Ok, let’s define the enumeration object.
Define enumerations and objects
GenderIdCodeEnum Defines our enumeration class GenderIdCodeEnum with id and code attributes:
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(a) {
return code;
}
@Override
public Integer getId(a) {
returnid; }}Copy the code
The requirements of this enumerated class are the same as above, if not clear, you can look again.
GenderIdCodeRequestBody: Request body for receiving JSON data
@Data
public class GenderIdCodeRequestBody {
private String name;
private GenderIdCodeEnum gender;
private long timestamp;
}
Copy the code
With the exception of the GenderIdCodeEnum parameter, everything is an example, so define it arbitrarily.
Implementing transformation logic
The prelude is set, and now it’s time to get down to business. Jackson offers two options:
- Scheme 1: Precision attack: Specifies the fields to be converted without affecting the fields of other objects
- Scheme 2: full range attack, all enumeration fields deserialized by Jackson, all have automatic conversion function
Plan 1: Precision attack
In this scenario, we first need to implement the JsonDeserialize abstract class:
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
finalClass<? > 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); }}}Copy the code
Then define the @jsondeserialize annotation on the specified enumeration field, such as:
@JsonDeserialize(using = IdCodeToEnumDeserializer.class)
private GenderIdCodeEnum gender;
Copy the code
To specify what each line does:
- Gets parameter values. Depending on your needs, this could be ID, code, or name, which is the source value that needs to be converted to an enumeration;
- Get the conversion guideline, which is prepared for steps 3 and 4;
- Get mark
@JsonDeserialize
Annotated field at this timecurrentName
The value isgender
; - Gets the wrapper object, which is
GenderIdCodeRequestBody
Object; - According to the wrapper object
Class
Object, and the field namegender
To obtainField
Object in preparation for Step 5; - To obtain
gender
The enumeration type corresponding to the field, i.eGenderIdCodeEnum
. The reason for doing this is to implement a generic deserialization class; - Here’s an implementation of write dead, where in an enumerated class you need to define a static method with the name
create
, the request parameter isObject
; - Call by reflection
create
Method to pass in the request parameters obtained in the first step.
Let’s look at the create method defined in the enumeration class:
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)) {
returnitem; }}return null;
}
Copy the code
For the sake of performance, we can define three sets of maps in advance, with id, code, name as key, and enumeration value as value, so that the time complexity of O(1) can be returned. Refer to the Converter class implementation logic above.
In this way, we can achieve precise conversion.
Plan 2: Full range attack
This scheme is a full-scope attack, and any deserialization that Jackson participates in, which has target enumeration parameters, will be subject to this logic. The solution is to define a static conversion method in an enumerated class, annotated with the @JsonCreator annotation, and Jackson will automatically convert.
The definition of this method is exactly the same as that of the create method in scenario 1, so you just need to annotate the create method:
@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)) {
returnitem; }}return null;
}
Copy the code
The Mode class has four values: DEFAULT, DELEGATING, PROPERTIES, and DISABLED, the differences of which will be explained in the principles section. In the same words, for application technology, we can first know the is, then know the why, but also must know the why.
test
Define a Controller method:
@PostMapping("gender-id-code-request-body")
public GenderIdCodeRequestBody bodyGenderIdCode(@RequestBody GenderIdCodeRequestBody genderRequest) {
genderRequest.setTimestamp(System.currentTimeMillis());
return genderRequest;
}
Copy the code
Then define the test case, again with 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\ : \" Look at the mountain \"}")
)
.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("Mountain", genderRequest.getName());
Assertions.assertTrue(genderRequest.getTimestamp() > 0);
}
Copy the code
At the end of the article to summarize
This article shows you how to gracefully use enumeration parameters in the RequestBody and customize the conversion logic with Jackson’s deserialization extension. Due to the length of the article, there are no large sections of code. Follow the public number “mountain hut” reply spring can obtain the source code. Follow me. Next we’ll move on to Principles.
Recommended reading
- SpringBoot: elegant response to achieve results in one move
- SpringBoot: How to handle exceptions gracefully
- SpringBoot: Dynamically injects the ID generator through the BeanPostProcessor
- SpringBoot: Customizes Filter to gracefully obtain request parameters and response results
- SpringBoot: Elegant use of enumeration parameters
- SpringBoot: Elegant use of enumeration parameters (Principles)
- SpringBoot: Gracefully use enumeration parameters in the RequestBody
- SpringBoot: Gracefully using enumeration parameters in the RequestBody
- Do unit tests with JUnit5+MockMvc+Mockito
- SpringBoot: Loads and reads resource files
Hello, I’m looking at the mountains. Swim in the code, play to enjoy life. If this article is helpful to you, please like, bookmark, follow. Welcome to follow the public account “Mountain Hut”, discover a different world.