background

When SpringCloud framework is selected to build micro-services for business background applications, a large number of business state value definitions will be involved. The general practice is as follows:

  • The persistence layer (database) stores values of type int
  • The background system uses a more readable constant to map values of int type
  • The front end (app or browser) also defines a set of constants to map these relationships
  • When the front end calls the interface of the background system, it commits with the int type defined by the constant

Due to optimization rules for persistent storage, ints are much more efficient than VARCHar types, and this approach is very well accepted.

However, there is a inconvenient point here: the constant definition of the state value mapping involves both front-end and back-end parts, communication cost is one aspect, and if the state value changes, two groups of people need to modify it at the same time.

target

In guarantee the persistence layer int type storage status value premise, mainly considering the state of the business can be read the sexual problems and modifications of the problem, can read part of sexual problems might be solved by defining constants front end staff, but when interface debugging or direct use of type int, this part can read sexual problems still exist, many modifications of problems need to solve.

This recommended scheme:

  • The persistence layer (database) does not store values of the original int type, which remains unchanged
  • Background system uses enum to define service status. Different service status sets can be realized by multiple EnUms, which support internationalization
  • The front end displays the text content of enum internationalization
  • When the front end invokes the background system interface, it submits the text content with enum internationalization
  • The background receives the internationalized text content of enum, converts it into an int value, and stores it in the database

Advantages of the scheme:

  • The efficiency of the original design of the durable layer is not affected
  • The definition and mapping of service status are all concentrated in the background system. When the status value changes, you only need to modify it in the background
  • The content displayed in the front end and the content transmitted by the interface is better readable text, and supports internationalization

Disadvantages of the scheme:

  • When the background system stores and reads status values, it needs to use enum to convert status values
  • The content packets transmitted by communication are slightly larger than the original int type

Project practice

Practice the principle of

This practice plan mainly includes three parts:

  1. The Enum class uses Jackson for JSON serialization and deserialization
  2. Enum Messages internationalization processing of enumeration items
  3. The definition of Enum

Enum Custom serialization and deserialization

First define the Enum internationalization class, customize the Enum serialization and deserialization class, and use annotations @jsonSerialize, @jsondeserialize registered in Spring ObjectMapper

@JsonDeserialize(using = DescEnumDeserializer.class)
@JsonSerialize(using = DescEnumSerializer.class)
public interface I18NEnum {

    /** * gets the enumeration description **@return* /
    String getDesc(a);
}

Copy the code

Consider a custom serialization implementation:

/ * * *@author huangying
 */
public class DescEnumSerializer extends JsonSerializer<I18NEnum> {

    @Override
    public void serialize(I18NEnum value, JsonGenerator gen, SerializerProvider serializers) throws IOException {
        // Concatenate configuration file keys by class name + enumeration value name, all uppercase processing
        String key = value.getClass().getSimpleName() + "." + StringUtils.upperCase(value.toString());
        // I18NUtil is the internationalization processing utility classString data = I18NUtil.get(key, value.getDesc()); gen.writeString(data); }}Copy the code

Custom deserialization implementation:

/ * * *@author huangying
 */
public class DescEnumDeserializer extends JsonDeserializer<I18NEnum> {

    @Override
    public I18NEnum deserialize(JsonParser p, DeserializationContext ctx) throws IOException {
        JsonNode node = p.getCodec().readTree(p);
        Class enumCls = BeanUtils.findPropertyType(p.currentName(), p.getCurrentValue().getClass());
        List enumFields = EnumUtils.getEnumList(enumCls);
        String keyPrefix = enumCls.getSimpleName() + ".";
        for (Object enumField : enumFields) {
            I18NEnum i18NEnum = (I18NEnum) enumField;
            // I18NUtil is the internationalization processing utility class
            String data = I18NUtil.get(keyPrefix + StringUtils.upperCase(i18NEnum.toString()), i18NEnum.getDesc());
            if (node.asText().equals(data)) {
                returni18NEnum; }}throw new I18NEnumException("Enum: unknown enumeration type"); }}Copy the code

Customize a special exception to look more elegant:

/ * * *@author huangying
 */
public class I18NEnumException extends RuntimeException {

    public I18NEnumException(String message) {
        super(message); }}Copy the code
Internationalization processing utility classes

Properties \messages_zh_CN. Properties \messages_en.properties \messages_en.properties \messages_en.properties \messages_en.properties \messages_en.properties \messages_en.properties And according to the specific language, return information to complete the internationalization display, the code is as follows:

/ * * *@author huangying
 */
@Component
public class I18NUtil {

    private static MessageSource messageSource;

    public I18NUtil(MessageSource messageSource) {
        I18NUtil.messageSource = messageSource;
    }

    public static String get(String key) {
        return messageSource.getMessage(key, null, LocaleContextHolder.getLocale());
    }

    public static String get(String key, Object arg) {
        return messageSource.getMessage(key, newObject[]{arg}, LocaleContextHolder.getLocale()); }}Copy the code
Enum Definition Example

Let’s take an example of an enum definition with two enumerations: SUCCESS and FAIL. The int value stored in the database is 1 and 2, respectively:

public enum OperateEnum implements I18NEnum {

	/** * Personal daily consumption */
	SUCCESS(1."SUCCESS"),
	/** ** */
	FAIL(2."FAIL");

	private int index;
	private String desc;

	OperateEnum(int index, String desc) {
		this.index = index;
		this.desc = desc;
	}

	@Override
	public String getDesc(a) {
		return desc;
	}

	public int getIndex(a) {
		returnindex; }}Copy the code

Write the configuration file as follows:

SUCCESS= SUCCESS OperateEnum.FAIL= FAIL # messages_zh_CN. Properties contents # enum class OperateEnum.SUCCESS= The operation succeeds OperateEnum.FAIL= The operation failsCopy the code

Application package

In the SpringCloud environment, we added the processing of international language, we unified the international language identifier in the request header lang:

/ * * *@author huangying
 */
public class I18NLocalResolver implements LocaleResolver {

    @Override
    public Locale resolveLocale(HttpServletRequest request) {
        String lang = request.getHeader("lang");
        // Get the DEFAULT LOCALE of the JVM
        Locale locale = Locale.getDefault();
        if(lang ! =null) {
            locale = new Locale(lang);
        }
        return locale;
    }

    @Override
    public void setLocale(HttpServletRequest request, HttpServletResponse response, Locale locale) {}}Copy the code

Custom enum serialization method trigger

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *

@apiOperation (value = "example of enumeration value internationalization ")
@APIIMPLICITParams ({@APIIMPLICITParam (name = "uid", value = "operator ID", paramType = "header", dataType = "Long")})
@RequestMapping(value = "/test/enums", method = RequestMethod.GET)
public Result get(
		@RequestHeader(value = "lang") String lang) {
	return Result.success(EnumUtils.getEnumList(OperateEnum.class));
}
Copy the code

The deserialization method of a custom enum is triggered

MappingJackson2HttpMessageConverter converter will default to the content of the @ RequestBody do deserialization processing, if the internationalization of enum value passed to the client, if need to deal with the client to submit content enumerated values of internationalization, The easiest way to do this is to define an enum in an @RequestBody object, which automatically triggers the custom deserialization method of the enum and gets the desired result.

If you define an enum object on the @requestParam modifier, In the request String into a enum is through org. Springframework. Core. The convert. Support. StringToEnumConverterFactory, This Class implements the interface ConverterFactory, which does this by calling Enum. ValueOf (Class, String) without triggering deserialization of Enum enumerated values. Therefore, only literal values (name) that are the same as enumerated values can be processed. Enum enum values may not be the same as literal values after internationalization.

If you want @RequestParam to trigger deserialization of enum enum values, you can try overriding the SpringMVC parameter converter, which is omitted here.

summary

The internationalization of enum enum values is a very interesting improvement, which may solve the problem of readability and improve the cohesion of business definitions. The application of this solution depends on the coding habits of the front and back end. If it is in the early stage of the project, you can try this solution after the communication and confirmation of the front and back end.

Focus on Java high concurrency, distributed architecture, more technology dry products to share and experience, please pay attention to the public account: Java architecture community can scan the left TWO-DIMENSIONAL code to add friends, invite you to join the Java architecture community wechat group to discuss technology! [Java Architecture Community]

(p1-jj.byteimg.com/tos-cn-i-t2…).