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

spring?mybatis環(huán)境常量與枚舉轉(zhuǎn)換示例詳解

 更新時(shí)間:2023年06月02日 08:49:10   作者:Mzoro  
這篇文章主要為大家介紹了spring?mybatis環(huán)境常量與枚舉轉(zhuǎn)換示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

常量與枚舉對象的轉(zhuǎn)換

日常 web 接口開發(fā)中經(jīng)常遇到一些常量與枚舉對象的轉(zhuǎn)換,如:請求中 sex 值為 1, 對應(yīng) java 程序中的枚舉為 Sex.BOY。如果要進(jìn)行值的比較,對于 java 開發(fā),枚舉最方便的比較方式就是使用 == 直接進(jìn)行比較,這時(shí)就要將 sex 轉(zhuǎn)為枚舉,然后再進(jìn)行后序判斷。

而大多數(shù)程序中的枚舉不是一個(gè)兩個(gè),可能上百上千個(gè)。用到以上處理邏輯的代碼會(huì)更多。那么是否可以將這一邏輯拆分到業(yè)務(wù)代碼之外,但又不改變接口調(diào)用方傳值,也不改變原有響應(yīng)值的形式呢?

另一種情況與前面提到的情況類似。大多數(shù) web 程序中的數(shù)據(jù)最終會(huì)寫入數(shù)據(jù)庫,有些 DB 設(shè)計(jì)對存儲要求很嚴(yán)格,一些標(biāo)識類字段希望占用空間越少越好。在上面的情況中 Sex.BOY, 在數(shù)據(jù)庫中希望記錄為 1 而不是 BOY。如果使用 mybatis ,這可能需要實(shí)現(xiàn)大量的 TypeHandler 接口進(jìn)行數(shù)據(jù)轉(zhuǎn)換與逆向轉(zhuǎn)換。是否有辦法自動(dòng)這一過程呢?

下面分別解決上面兩個(gè)需求。這兩個(gè)需求的實(shí)現(xiàn)都依賴類似下面這樣的一個(gè)接口。

public interface CodeData {
    String getCode();
}

接口這樣設(shè)計(jì)是因?yàn)橐陨闲枨蟪橄笠院?,都是將一個(gè)編碼在合適的時(shí)機(jī)轉(zhuǎn)化成一個(gè)枚舉或者將一個(gè)枚舉轉(zhuǎn)為編碼。所以問題的關(guān)鍵只是一個(gè)編碼,至于編碼對應(yīng)的內(nèi)容,是給人看的,而不是為了計(jì)算機(jī)。

假設(shè) Sex 枚舉如下定義

public enum Sex implements CodeData{
    BOY("1","男"),
    GIRL("2", "女");
    public final String code;
    public final String text;
    public(String code, String text){
        this.code = code;
        this.text = text;
    }
    @Override
    public String getCode(){
        return this.code;
    }
    public String getText(){
        return this.text;
    }
}

為方便說明其使用方式,再簡單定義一個(gè) User 類型

public Class User{
    private String id;
    private String name;
    private Sex sex;
    // 又臭又長的getter/setter , 一定要有
}

對接口的請求與響應(yīng)數(shù)據(jù)進(jìn)行進(jìn)行自動(dòng)轉(zhuǎn)換

通常我們寫一個(gè) @RestController 接口, 方法定義的參數(shù)大概分為以下兩種情況:

  • 使用 @RequestBody 修飾參數(shù)
  • 不使用 @RequestBody 修飾

當(dāng)然還有其他的情況,暫不涉及。

使用 @RequestBody 的情況

當(dāng)使用 @RequestBody 修飾時(shí),方法定義大概如下:

public User create(@RequestBody User user){
    // ....
    return user;
}

如果你使用了 springboot 那更方便了,默認(rèn)情況下(沒有對 pom 進(jìn)行大的改動(dòng),排除 jackson 依賴),springboot 向工程中注入了 Jackson2ObjectMapperBuilderCustomizer ^2 用于方便用戶修改對 json 類型的請求體進(jìn)行反序列化。我們要做的就是告訴 spring 如何對 Sex 這類的屬性進(jìn)行反序列化。

這里需要用到的知識點(diǎn)是 jackson 的自定義序列化與反序列化 ^3。

需要針對 Sex 實(shí)現(xiàn) JsonDeserializer

public class SexDeserializer extends JsonDeserializer<Sex> {
    @Override
    public Sex deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {
        String text = jsonParser.getText();
        if (StringUtils.isBlank(text)) {
            return null;
        }
        for (Sex e : Sex.values()) {
            if (text.equals(e.getCode())) {
                return e;
            }
        }
        return null;
    }
}

然后將這個(gè) Deserializer 注入到 spring 環(huán)境中的 ObjectMapper 中

@Configuration
public class CustomJsonSerializerConfig{
    @Bean
    public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer(CustomJsonSerializerProvider provider) {
        return jacksonObjectMapperBuilder -> {
            jacksonObjectMapperBuilder.deserializerByType(Sex.class, new SexDeserializer());
        }
    }
}

這樣對 Sex 類型的屬性反序列化都會(huì)由 SexDeserializer

下一個(gè)問題是,工程中有大量的枚舉需要做類似的處理,有沒有辦法自動(dòng)注冊這些類似的 Deserializer? 答案是肯定的, 編程不解決類似的,重復(fù)性的工作,就不要編程了。

要完成自動(dòng)向環(huán)境中注冊, 需要以下兩個(gè)前提

  • 掃描到 classpath 下所有需要自定義反序列化的類型
  • 對每個(gè)類型構(gòu)造一個(gè) Deserializer

第一個(gè)問題特別復(fù)雜,需要篇幅很長,不能在本文仔細(xì)說明, 涉及到自定義 ClassPathBeanDefinitionScanner, 需要理解 BeanDefinitionRegistryPostProcessorBeanDefinition 以及 springboot 啟動(dòng)的大致步驟??梢詤⒖?mybatis 如何自動(dòng)構(gòu)造 mapper:org.mybatis.spring.boot.autoconfigure.MybatisAutoConfiguration,org.mybatis.spring.mapper.ClassPathMapperScanner。

掃描類時(shí),我們需要收集所有在 classpath 下的 CodeData 的枚舉類型的實(shí)現(xiàn)類, 將這些枚舉類型的集合注入到 spring 環(huán)境中。我們構(gòu)造下面這樣的一個(gè)類完成這件事。

public record CodeDataTypeProvider<T extends CodeData>(Set<T> codeDataTypes){}

解決第一個(gè)問題后,下一個(gè)問題就是為每一個(gè) CodeData 類型構(gòu)造一個(gè) Deserializer 對象,我們需要設(shè)計(jì)一個(gè)抽象程序更高的類,這樣就不用為第一個(gè) CodeData 實(shí)現(xiàn)類創(chuàng)建一個(gè)特別的 Deserializer 了:

public class CodeDataDeserializer<T extends CodeData> extends JsonDeserializer<T> {
    private final T[] enums;
    public CodeDataDeserializer(Class<T> enumClass) {
        if (enumClass == null) {
            throw new IllegalArgumentException("Type argument cannot be null");
        }
        if (!enumClass.isEnum()) {
            throw new IllegalArgumentException(enumClass.getName() + "is not a enum");
        }
        this.enums = enumClass.getEnumConstants();
    }
    @Override
    public T deserialize(JsonParser jsonParser, DeserializationContext deserializationContext) throws IOException {
        String text = jsonParser.getText();
        if (StringUtils.isBlank(text)) {
            return null;
        }
        for (T e : enums) {
            if (text.equals(e.getCode())) {
                return e;
            }
        }
        return null;
    }
}

最后一步就是將這些 Deserializer 注冊到環(huán)境中的 ObjectMapper

@Bean
    public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer(CustomJsonSerializerProvider provider) {
        return jacksonObjectMapperBuilder -> {
            Set<Class<CodeData>> codeDataClass = provider.codeDataType();
            if (CollectionUtils.isEmpty(codeDataClass)) {
                logger.warn("There is no JsonSerializer and CodeData type.");
                return;
            }
            if (codeDataClass != null) {
                for (Class<? extends CodeData> c : codeDataClass) {
                    logger.debug("register custom json serializer {} for {}", "CodeDataSerializer", c);
                    jacksonObjectMapperBuilder.deserializerByType(c, new CodeDataDeserializer<>(c));
                    // 這一步是為了響應(yīng)數(shù)據(jù)時(shí),也能自動(dòng)將CodeData按同樣的規(guī)則序列化
                    jacksonObjectMapperBuilder.serializerByType(c, new CodeDataSerializer<>());
                }
            }
        };
    }

不使用 @RequestBody 的情況

這種情況下,方法的定義通常是下面的樣子

@GetMapping("/getUserBySex")
public SearchUserResultBo getUserBySex(Sex sex){}
@GetMapping("/getUserBySex")
public SearchUserResultBo getUserBySex(@RequestParam Sex sex){}
@GetMapping("/getUserBySex/{sex}")
public SearchUserResultBo getUserBySex(@PathVariable Sex sex){}
@GetMapping("/getUser")
public User getUser(User user){}

這種情況情況下 spring 為我們預(yù)留了 org.springframework.format.Formatter 這個(gè)接口, 它的用法與 jackson 稱之為序列化類型的用法類似, 為需要自定義轉(zhuǎn)換的類型構(gòu)造一個(gè) Formatter , 注冊到工程環(huán)境中。

同樣存在前面的問題, Formatter 數(shù)量多且邏輯重復(fù)。 但是有前面基礎(chǔ),這里就比較輕松了, 共通的 Formatter 如下:

public class CodeDataFormatter<T extends CodeData> implements Formatter<T> {
    private final Class<T> tClass;
    public CodeDataFormatter(Class<T> tClass) {
        if (tClass == null) {
            throw new IllegalArgumentException("class can not be null.");
        }
        if (!tClass.isEnum()) {
            throw new IllegalArgumentException("class can not be null and must be a enum :" + tClass.getName());
        }
        this.tClass = tClass;
    }
    @Override
    public T parse(@Nonnull String text, @Nonnull Locale locale) throws ParseException {
        if (StringUtils.isBlank(text)) {
            return null;
        }
        T[] values = tClass.getEnumConstants();
        for (T t : values) {
            if (StringUtils.equal(t.getCode(), text)) {
                return t;
            }
        }
        return null;
    }
    @Override
    public String print(T object, Locale locale) {
        if (object == null) {
            return null;
        }
        return object.getCode();
    }
}

注冊到環(huán)境中的方式如下

@Configuration
public class WebConfig implements WebMvcConfigurer {
    @SuppressWarnings("unchecked")
    @Override
    public void addFormatters(FormatterRegistry registry) {
        Set<Class<CodeData>> codeDataClass = provider.codeDataType();
        if (CollectionUtils.isEmpty(codeDataClass)) {
            logger.warn("There is no CodeData type.");
            return;
        }
        for (Class<? extends CodeData> c : codeDataClass) {
            logger.debug("register formatter {} for {}", "CodeDataFormatter", c);
            registry.addFormatterForFieldType(c, new CodeDataFormatter<>(c));
        }
    }
}

通過 mybatis 將數(shù)據(jù)落庫與讀取

對于 mybatis 的處理與 @RequestBody 的處理方式類似, 使用 spring boot 的工程都會(huì)通過 mybatis-spring-boot-starter 引入 mybatis , 這個(gè)依賴引入后,程序啟動(dòng)時(shí),mybatis autoconfig 會(huì)獲取環(huán)境中的 ConfigurationCustomizer 對 mybatis 配置進(jìn)行定制化, 所以我們環(huán)境中注入一個(gè) ConfigurationCustomizer 就可以了

   @Bean
    ConfigurationCustomizer mybatisConfigurationCustomizer(MybatisTypeHandlerProvider provider) {
        return configuration -> {
            // customize ...
            Set<Class<CodeData>> codeDataClass = provider.codeDataClass();
            if (CollectionUtils.isEmpty(codeDataClass)) {
                logger.warn("There is no TypeHandler and CodeData type.");
                return;
            }
            TypeHandlerRegistry registry = configuration.getTypeHandlerRegistry();
            if (codeDataClass != null) {
                for (Class<CodeData> c : codeDataClass) {
                    logger.debug("register type handler {} for {}", "CodeDataTypeHandler", c);
                    registry.register(c, new CodeDataTypeHandler<>(c));
                }
            }
        };
    }

以上就是spring mybatis環(huán)境常量與枚舉轉(zhuǎn)換示例詳解的詳細(xì)內(nèi)容,更多關(guān)于spring mybatis環(huán)境常量枚舉轉(zhuǎn)換的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

最新評論