关于后端的数据字典的实现方案

方案一:使用数据库完成查询数据返回给前端

这种方案有点是灵活性高,可以直接操作数据库实现字段的改变,可动态修改,但是问题就是比较依赖数据库,如果数据库数据丢失或者是数据库挂掉以后数据会丢失,并且从数据的效率来说需要通过连接池拿取链接再查询数据库,不是很友好,但是可以通过redis来优化,总体而言也是一种不错的解决方案,适用于数据字典比较大的情况

方案二:类似于第一种方案但是是基于nacos动态配置实现

nacos是可以配置map的数据结构的,可以通过nacos的动态配置功能实现动态修改数据字典以后返回给前端,优点是和数据库的方法一样可以动态修改,但是相对于数据可而言又可以直接在网页操作,可视化更高,使用更加方便,但是优点也很明显,就是基于nacos,同样存在nacos挂掉之类的问题,这种方案适用于数据字典很小的情况(数据字典太多会让配置文件不雅观),并且本来项目中就需要nacos并且支持动态加载(如果只是单纯因为想用数据字典而引用nacos并且配置动态加载的话是有点杀鸡用牛刀的感觉,没有太大的必要)

方案三:使用反射的方法来获取数据字典

这种方案的话优点是可以不借助数据库或者是nacos的第三方软件,性能上来说会大于前两者,但是不支持灵活配置,如果项目中的数据字典大部分是由枚举构成,又想不集成第三方的时候可以考虑使用,并且对于数据字典的灵活性要求不高

以上三种方案均可简单实现数据字典的方法,主要说一下第二种和第三种的实现方法,第一种和普通的数据库查询没区别就不做讲述

实现

nacos实现--(我这里是基于spring-cloud做的)

第一步需要在项目种集成nacos

       
            com.alibaba.cloud
            spring-cloud-starter-alibaba-nacos-config
        

第二步需要配yml文件配置我们需要的对象

dictionary: '{"bbb": "", "ccc": "", "aaa": ""}'

bbb表示你的k ""表示的是v。根据业务场景v也可以是其他数据格式

第三步 在控制层使用@Value实现读取yml中的配置信息

@RefreshScope
public class CommonController {
    @Value("#{${customer-service}}")
    private Map dictionary;

@RefreshScope注解根据实际情况看是否配置,这个注解是表示这个配置支持动态加载,可以实现动态修改数据字典的功能,使用map接受的时候注意value中的书写格式否则读取不到

接下来就是将字典返回给前端了

反射实现--(我这是通过前端传入枚举返回具体的枚举对象用于做下拉框之类的操作)

第一步创建枚举

public enum TopicType {

    RADIO("radio", "单选题"),

    MULTIPLE_CHOICE("multipleChoice", "多选题"),

    DETERMINE("determine", "判断题"),

    UNDIRECTED_SELECTION("undirectedSelection", "不定向选择"),

    PACK("pack", "填空题");

    private final String value;

    private final String name;
}

我们这里统一value做英文,name做中文方便后面的反射使用

第二步通过反射创建对象然后通过反射方法调用获取枚举种所有对象

 String classpath = "com.memoo.common.core.enums.kl.TopicType ";
  Object[] menus;
 Class aClass = Class.forName(classpath);
 Method values = aClass.getMethod("values");
 menus = (Object[]) values.invoke(null);

这个时候返回的数组对象就是我们的整个枚举对象了

第三步通过反射调用获取枚举的name和value

 for (Object me : menus) {
                Method getName = aClass.getMethod("getName");
                Object name = getName.invoke(me, null);
                System.out.println(name);
                Method getValue = aClass.getMethod("getValue");
                Object value = getValue.invoke(me, null);
                System.out.println(value);
            }

再通过getName和getValue方法返回具体的显示name和value

   public static void main(String[] args) {
        String classpath = "com.memoo.common.core.enums.kl.SubjectType";
        Object[] menus;
        HashMap objectObjectHashMap = new HashMap<>();
        try {
            Class aClass = Class.forName(classpath);
            Method values = aClass.getMethod("values");
            menus = (Object[]) values.invoke(null);
            for (Object me : menus) {
                Method getName = aClass.getMethod("getName");
                Object name = getName.invoke(me, null);
                System.out.println(name);
                Method getValue = aClass.getMethod("getValue");
                Object value = getValue.invoke(me, null);
                System.out.println(value);
            }

        } catch (Exception e) {
        }
    }

输出结果

练习
practice
速记
shorthand

可配合前端将SubjectType当作变量由前端传入,从而可以通过接口获取下拉框信息

ps:有什么补充或者独特见解可随时沟通

你可能感兴趣的:(关于后端的数据字典的实现方案)