基于Springboot一个注解搞定数据字典的实践方案
作者:我不在线223 发布时间:2022-12-23 01:12:38
标签: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 的中文值。
来源:https://juejin.cn/post/7104530636794232862


猜你喜欢
- 摘要:本文演示如何构建起一个优秀的后端接口体系,体系构建好了自然就有了规范,同时再构建新的后端接口也会十分轻松。一个后端接口大致分为四个部分
- 说明使用工具:brew caskbrew cask是一个用命令行管理Mac下应用的工具,提供了自动安装和卸载功能,能够自动从官网上下载并安装
- 启动Activity并传递参数Extra正常情况下启动Activity并且传递参数的代码:Intent intent = new Inten
- 1. Ajax 概述Ajax 的英文全称是 ”Asynchronous JavaScript and XML&l
- 使用范围synchronized使用上用于同步方法或者同步代码块在锁实现上是基于对象去实现使用中用于对static修饰的便是class类锁使
- Linux+Docker+SpringBoot+IDEA一键自动化部署的步骤记录从打包到服务器配置上线全流程安装docker详细步骤请戳这里
- 当我们需要制作动态炫酷科技感很强的UI时,美术一般会给我们提供一些序列图,这时候我们只需在程序里实现序列动画。一.动画机unity自带的帧动
- 一、什么是iText?在企业的信息系统中,报表处理一直占比较重要的作用,iText是一种生成PDF报表的Java组件。通过在服务器端使用Js
- 本文实例讲述了C#创建、读取和修改Excel的方法。分享给大家供大家参考。具体如下:windows下我们可以通过 Jet OLE DB访问E
- 引言 在多线程中,为了使线程安全,我们经常会使用synchronized和Lock进行代码同步和加锁,但是具体两者有什么区别,什
- 可能经常看面经的同学都知道,面试所遇到的排序算法,快速排序占主要位置,热度只增不减啊,其次就是归并和堆排序。其实以前写过一篇排序的文章,写的
- 概述一天之计在于晨,每天的早餐也是必不可少,但是很多人为了节约时间,都是简单的吃点凑合一下或干脆不吃早餐,这对于个人身体和工作效率来说,无疑
- 作为一个ORM框架,hibernate肯定也需要满足我们实现表与表之间进行关联的需要。hibernate在关联方法的实现很简单。下面我们先来
- 如下所示:public static void main(String[] args) {String str1 = "刘烨,孙坚
- 本篇文章主要介绍了Android 三方库混淆规则,分享给大家,具体如下:基本指令-optimizationpasses 5-dontusem
- c#数据绑定之将datatabel的data添加listView中,简要的通过代码应用了DataTable,DataTableColumns
- 使用AES算法可用于对数据进行加密码与解密,使用的时候需要注意两点:1)被加密的串越长,加密后的字符串越长,注意数据库字段的设计;2)Lin
- 一、什么是相对布局相对布局是另外一种控件摆放的方式相对布局是通过指定当前控件与兄弟控件或者父控件之间的相对位置,从而达到相对的位置二、为什么
- java 读取网页内容的实例详解import java.io.BufferedReader; import java.io.IOExcept
- Kotlin 支持泛型, 语法和 Java 类似。例如,泛型类:class Hello<T>(val value: T)val