首頁 > 軟體

基於Springboot一個註解搞定資料字典的實踐方案

2022-06-10 18:00:47

問題引出:

最近開了新專案,專案中用到了資料字典,列表查詢資料返回的時候需要手動將code轉換為name,到前臺展示。專案經理表示可以封裝一個統一的功能,避免程式設計師各自寫各自的,程式碼混亂,風格不統一。

要求:

  • 基於微服務架構,資料字典通過服務獲取;
  • 簡化程式碼,使用簡單;
  • 使用Redis;

方案

大致的方向是自定義註解,在序列化的時候進行資料處理; 考慮到微服務,需要將主要邏輯放到common中,然後對外提供介面,各業務服務實現介面以獲取字典資料; 考慮Redis,序列化處理資料時,首先通過Redis獲取,獲取不到在通過介面獲取,拿到資料後存到Redis中,然後再返回處理; 也可以多做一步,在新增、修改資料字典時,同步更新Redis內容,以保證資料有效性。

實現

  • 定義註解
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@JacksonAnnotationsInside
@JsonSerialize(using = DictSerializer.class)
public @interface Dict {

    /** 字典型別 */
    String type();
}
  • 指定註解新增位置
  • 指定註解生效時間
  • 指定序列化處理類
  • 序列化處理類
public class DictSerializer extends StdSerializer<Object> implements ContextualSerializer {
    /** 字典註解 */
    private Dict dict;
    public DictSerializer() {
        super(Object.class);
    }
    public DictSerializer(Dict dict) {
        super(Object.class);
        this.dict = dict;
    }
    private String type;
    @Override
    public void serialize(Object value, JsonGenerator gen, SerializerProvider provider) throws IOException {
        if (Objects.isNull(value)) {
            gen.writeObject(value);
            return;
        }
        if (Objects.nonNull(dict)){
            type = dict.type();
        }
        // 通過資料字典型別和value獲取name

        gen.writeObject(value);
        gen.writeFieldName(gen.getOutputContext().getCurrentName()+"Name");
        gen.writeObject(label);
    }
    @Override
    public JsonSerializer<?> createContextual(SerializerProvider prov, BeanProperty beanProperty) throws JsonMappingException {
        if (Objects.isNull(beanProperty)){
            return prov.findValueSerializer(beanProperty.getType(), beanProperty);
        }
        Dict dict = beanProperty.getAnnotation(Dict.class);
        if (Objects.nonNull(dict)){
            type = dict.type();
            return this;
        }
        return prov.findNullValueSerializer(null);
    }
}

這裡處理的邏輯是原先的欄位內容不變,新增一個新的欄位用來儲存轉化後的值;

  • 資料字典獲取
private static String changeLabel(String type,String code) {
    if(code.indexOf(",") > -1) {
        String[] strs = code.split(",");
        if (strs.length > 1) {
            StringBuilder sb = new StringBuilder();
            for (String str : strs) {
                // 從快取中獲取字典。如果不行,通過SpringUtil.getBean(); 獲取服務處理
                sb.append(DictDataCache.getLabel(type, str)).append(separator);
            }
            return sb.substring(0, sb.length() - 1);
        }
    }
    // 從快取中獲取字典。如果不行,通過SpringUtil.getBean(); 獲取服務處理
    return DictDataCache.getLabel(type, code);
}

考慮存在多選的情況,先判斷下是否是多選的,預設逗號拼接,後期新增入參控制;

@Override
public String getDictDataOptions(String typeCode,String value) {
    if (redisTemplate.hasKey("dict:"+typeCode+":"+value)){
        return (String) redisTemplate.opsForValue().get("dict:"+typeCode+":"+value);
    }
    List<DictDataOptions> dictDataList = getDictDataHandler().getDictDataOptions(typeCode);
    if(CollUtil.isNotEmpty(dictDataList)) {
        put(typeCode, dictDataList);
    }
    if (redisTemplate.hasKey("dict:"+typeCode+":"+value)){
        return (String) redisTemplate.opsForValue().get("dict:"+typeCode+":"+value);
    }
    return null;
}

根據key判斷Redis中是否存在,存在則直接獲取,不存在則通過介面獲取,獲取到直接放到Redis中,然後再次從Redis獲取。

protected void put(String typeCode, List<DictDataOptions> dataList) {
    if (CollUtil.isNotEmpty(dataList)){
        for (DictDataOptions dictDataOptions : dataList) {
            AbstractDictHandler.redisTemplate.opsForValue().set("dict:"+typeCode+":"+dictDataOptions.getDataLabel(),dictDataOptions.getDataValue());
        }
    }
}

迴圈放置資料字典值

@Override
public List<DictDataOptions> getDictDataOptions(String typeCode) {
    return iSysDictService.queryDictItemsByCode(typeCode).stream()
            .map(e -> DictDataOptions.builder().typeCode(typeCode).dataLabel(e.getValue()).dataValue(e.getText()).build())
            .collect(Collectors.toList());
}

根據資料字典型別,通過介面獲取資料;注意該實現類需要每個微服務實現一個;然後為了避免基礎資料服務掛掉,呼叫報錯,common中提供一個預設實現。

4.使用

@Dict(type = "inspectType")
private String checkType;

在返回前端的實體中,對應欄位新增註解,並指定資料字典type值

      {
        "id": "1522492702905954306",
        "professionName": "專業名稱888",
        "checkCode": "檢測項編碼8",
        "checkProject": "rrrr檢測專案88",
        "checkDevice": "52",
        "checkStandard": "檢測項編碼88",
        "referenceStandard": "wq參考標準8",
        "checkType": "1",
        "checkTypeName": "尺寸",
        "remarks": "ef備註備註8"
      },

前端獲取的json會多一個欄位:checkTypeName,內容為checkType 的中文值。

到此這篇關於基於Springboot一個註解搞定資料字典問題的文章就介紹到這了,更多相關Springboot資料字典內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com