基於Springboot一個註解搞定數據字典的實踐方案
問題引出:
最近開瞭新項目,項目中用到瞭數據字典,列表查詢數據返回的時候需要手動將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數據字典內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- 使用@Autowired 註入RedisTemplate報錯的問題及解決
- Springboot/Springcloud項目集成redis進行存取的過程解析
- SpringBoot集成Redis實現驗證碼的簡單案例
- springboot使用redis的詳細步驟
- SpringBoot詳解如何整合Redis緩存驗證碼