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

SpringBoot中實現(xiàn)數(shù)據(jù)字典的示例代碼

 更新時間:2020年09月10日 11:20:25   作者:Asurplus、  
這篇文章主要介紹了SpringBoot中實現(xiàn)數(shù)據(jù)字典的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

我們在日常的項目開發(fā)中,對于數(shù)據(jù)字典肯定不模糊,它幫助了我們更加方便快捷地進(jìn)行開發(fā),下面一起來看看在 SpringBoot 中如何實現(xiàn)數(shù)據(jù)字典功能的

一、簡介

1、定義

數(shù)據(jù)字典是指對數(shù)據(jù)的數(shù)據(jù)項、數(shù)據(jù)結(jié)構(gòu)、數(shù)據(jù)流、數(shù)據(jù)存儲、處理邏輯等進(jìn)行定義和描述,其目的是對數(shù)據(jù)流程圖中的各個元素做出詳細(xì)的說明,使用數(shù)據(jù)字典為簡單的建模項目。簡而言之,數(shù)據(jù)字典是描述數(shù)據(jù)的信息集合,是對系統(tǒng)中使用的所有數(shù)據(jù)元素的定義的集合。

數(shù)據(jù)字典(Data dictionary)是一種用戶可以訪問的記錄數(shù)據(jù)庫和應(yīng)用程序元數(shù)據(jù)的目錄。主動數(shù)據(jù)字典是指在對數(shù)據(jù)庫或應(yīng)用程序結(jié)構(gòu)進(jìn)行修改時,其內(nèi)容可以由DBMS自動更新的數(shù)據(jù)字典。被動數(shù)據(jù)字典是指修改時必須手工更新其內(nèi)容的數(shù)據(jù)字典。

2、理解

數(shù)據(jù)字典是一種通用的程序設(shè)計思想,將主體與分支存于兩張數(shù)據(jù)表中,他們之間靠著唯一的 code 相互聯(lián)系,且 code 是唯一存在的,分支依附主體而存在,每一條分支都有它唯一對應(yīng)的屬性值

例如:性別(sex),分為(0–保密1–男2–女),那么數(shù)據(jù)字典的設(shè)計就應(yīng)該是

主表:

{
	"code": "sex",
	"name": "性別"
}

副表:

[{
		"dictCode": "sex",
		"code": "0",
		"text": "保密"
	},
	{
		"dictCode": "sex",
		"code": "1",
		"text": "男"
	},
	{
		"dictCode": "sex",
		"code": "2",
		"text": "女"
	}
]

那么我們在使用數(shù)據(jù)字典的時候,只需要知道 dictCode,再使用 code 找到唯一的字典值

二、數(shù)據(jù)表設(shè)計

1、數(shù)據(jù)表設(shè)計

主表:

drop table if exists sys_dict;

/*==============================================================*/
/* Table: sys_dict                       */
/*==============================================================*/
create table sys_dict
(
  id          bigint(20) not null auto_increment comment '主鍵id',
  code         varchar(32) comment '編碼',
  name         varchar(32) comment '名稱',
  descript       varchar(64) comment '描述',
  status        tinyint(1) default 0 comment '狀態(tài)(0--正常1--凍結(jié))',
  create_time     datetime comment '創(chuàng)建時間',
  create_user     bigint(20) comment '創(chuàng)建人',
  del_flag       tinyint(1) default 0 comment '刪除狀態(tài)(0,正常,1已刪除)',
  primary key (id)
)
type = InnoDB;

alter table sys_dict comment '字典管理表';

副表:

drop table if exists sys_dict_detail;

/*==============================================================*/
/* Table: sys_dict_detail                    */
/*==============================================================*/
create table sys_dict_detail
(
  id          bigint(20) not null comment '主鍵id',
  dict_code      varchar(32) comment '字典編碼',
  code         varchar(32) comment '編碼',
  name         varchar(32) comment '名稱',
  primary key (id)
)
type = InnoDB;

alter table sys_dict_detail comment '字典配置表';

它們的關(guān)系如圖所示:

2、數(shù)據(jù)字典配置

三、開發(fā)前戲

1、引入 maven 依賴

<!-- web支持 -->
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<!-- thymeleaf模板引擎 -->
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
<!-- aop依賴 -->
<dependency>
  <groupId>org.springframework.boot</groupId>
  <artifactId>spring-boot-starter-aop</artifactId>
</dependency>
<!-- lombok插件 -->
<dependency>
  <groupId>org.projectlombok</groupId>
  <artifactId>lombok</artifactId>
  <optional>true</optional>
</dependency>

我們引入了 aop 切面所需依賴,我們的數(shù)據(jù)字典也是基于 aop 切面實現(xiàn)的

2、創(chuàng)建實體類

用戶信息表 SysUserInfo.java:

import com.baomidou.mybatisplus.annotation.*;
import com.baomidou.mybatisplus.extension.activerecord.Model;
import com.zyxx.common.annotation.Dict;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;

import java.io.Serializable;

/**
 * <p>
 * 用戶信息表
 * </p>
 *
 * @author lizhou
 * @since 2020-07-06
 */
@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@TableName("sys_user_info")
@ApiModel(value="SysUserInfo對象", description="用戶信息表")
public class SysUserInfo extends Model<SysUserInfo> {


  @ApiModelProperty(value = "ID")
  @TableId(value = "id", type = IdType.AUTO)
  private Long id;

  @ApiModelProperty(value = "登錄賬號")
  @TableField("account")
  private String account;

  @ApiModelProperty(value = "登錄密碼")
  @TableField("password")
  private String password;

  @ApiModelProperty(value = "姓名")
  @TableField("name")
  private String name;

  @ApiModelProperty(value = "性別(0--未知1--男2--女)")
  @TableField("sex")
  @Dict(dictCode = "sex")
  private Integer sex;

  @ApiModelProperty(value = "狀態(tài)(0--正常1--凍結(jié))")
  @TableField("status")
  @Dict(dictCode = "status")
  private Integer status;
}

3、返回結(jié)果通用實體類

返回結(jié)果通用實體類 LayTableResult.java:

import lombok.Getter;
import lombok.Setter;

import java.util.List;

/**
 * @param <T> 返回的實體類
 * @author lizhou
 * @描述 后臺返回給LayUI的數(shù)據(jù)格式
 */
@Getter
@Setter
public class LayTableResult<T> {

  /**
   * 接口狀態(tài)
   */
  private Integer code;

  /**
   * 提示信息
   */
  private String msg;

  /**
   * 接口數(shù)據(jù)長度
   */
  private Long count;

  /**
   * 接口數(shù)據(jù)
   */
  private List<T> data;

  /**
   * 無參構(gòu)造函數(shù)
   */
  public LayTableResult() {
    super();
  }

  /**
   * 返回數(shù)據(jù)給表格
   */
  public LayTableResult(Long count, List<T> data) {
    super();
    this.count = count;
    this.data = data;
    this.code = 0;
  }
}

由于我用的是 layui 前端框架,我寫了一個返給 layui 表格的通用實體類,這是在實現(xiàn)數(shù)據(jù)字典需要用到的,判斷響應(yīng)返回實體類的類型來判斷是否需要注入字典

四、開發(fā)實現(xiàn)

1、創(chuàng)建自定義注解

我們創(chuàng)建一個自定義注解 @Dict 來實現(xiàn)數(shù)據(jù)字典

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 數(shù)據(jù)字典注解
 *
 * @author Tellsea
 * @date 2020/6/23
 */
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Dict {

  /**
   * 字典類型
   *
   * @return
   */
  String dictCode();

  /**
   * 返回屬性名
   *
   * @return
   */
  String dictText() default "";
}

2、注解實現(xiàn)

我們使用 aop 切面來實現(xiàn)什么的自定義注解 @Dict

import com.alibaba.fastjson.JSONObject;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.zyxx.common.annotation.Dict;
import com.zyxx.common.utils.LayTableResult;
import com.zyxx.common.utils.ObjConvertUtils;
import com.zyxx.sbm.entity.SysDictDetail;
import com.zyxx.sbm.service.SysDictService;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.lang.reflect.Field;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;

/**
 * 數(shù)據(jù)字典切面
 *
 * @author Tellsea
 * @date 2020/6/23
 */
@Aspect
@Component
@Slf4j
public class DictAspect {

	/**
	* 字典后綴
	*/
  private static String DICT_TEXT_SUFFIX = "Text";

  @Autowired
  private SysDictService sysDictService;

	/**
	* 切點(diǎn),切入 controller 包下面的所有方法
	*/
  @Pointcut("execution( * com.zyxx.*.controller.*.*(..))")
  public void dict() {

  }

  @Around("dict()")
  public Object doAround(ProceedingJoinPoint pjp) throws Throwable {
    long time1 = System.currentTimeMillis();
    Object result = pjp.proceed();
    long time2 = System.currentTimeMillis();
    log.debug("獲取JSON數(shù)據(jù) 耗時:" + (time2 - time1) + "ms");
    long start = System.currentTimeMillis();
    this.parseDictText(result);
    long end = System.currentTimeMillis();
    log.debug("解析注入JSON數(shù)據(jù) 耗時" + (end - start) + "ms");
    return result;
  }

  private void parseDictText(Object result) {
    if (result instanceof LayTableResult) {
      List<JSONObject> items = new ArrayList<>();
      LayTableResult rr = (LayTableResult) result;
      if (rr.getCount() > 0) {
        List<?> list = (List<?>) rr.getData();
        for (Object record : list) {
          ObjectMapper mapper = new ObjectMapper();
          String json = "{}";
          try {
            // 解決@JsonFormat注解解析不了的問題詳見SysAnnouncement類的@JsonFormat
            json = mapper.writeValueAsString(record);
          } catch (JsonProcessingException e) {
            log.error("Json解析失?。? + e);
          }
          JSONObject item = JSONObject.parseObject(json);
          // 解決繼承實體字段無法翻譯問題
          for (Field field : ObjConvertUtils.getAllFields(record)) {
            //解決繼承實體字段無法翻譯問題
            // 如果該屬性上面有@Dict注解,則進(jìn)行翻譯
            if (field.getAnnotation(Dict.class) != null) {
              // 拿到注解的dictDataSource屬性的值
              String dictType = field.getAnnotation(Dict.class).dictCode();
              // 拿到注解的dictText屬性的值
              String text = field.getAnnotation(Dict.class).dictText();
              //獲取當(dāng)前帶翻譯的值
              String key = String.valueOf(item.get(field.getName()));
              //翻譯字典值對應(yīng)的text值
              String textValue = translateDictValue(dictType, key);
              // DICT_TEXT_SUFFIX的值為,是默認(rèn)值:
              // public static final String DICT_TEXT_SUFFIX = "_dictText";
              log.debug("字典Val: " + textValue);
              log.debug("翻譯字典字段:" + field.getName() + DICT_TEXT_SUFFIX + ": " + textValue);
              //如果給了文本名
              if (!StringUtils.isBlank(text)) {
                item.put(text, textValue);
              } else {
                // 走默認(rèn)策略
                item.put(field.getName() + DICT_TEXT_SUFFIX, textValue);
              }
            }
            // date類型默認(rèn)轉(zhuǎn)換string格式化日期
            if ("java.util.Date".equals(field.getType().getName())
                && field.getAnnotation(JsonFormat.class) == null
                && item.get(field.getName()) != null) {
              SimpleDateFormat aDate = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
              item.put(field.getName(), aDate.format(new Date((Long) item.get(field.getName()))));
            }
          }
          items.add(item);
        }
        rr.setData(items);
      }
    }
  }

  /**
   * 翻譯字典文本
   *
   * @param dictType
   * @param key
   * @return
   */
  private String translateDictValue(String dictType, String key) {
    if (ObjConvertUtils.isEmpty(key)) {
      return null;
    }
    StringBuffer textValue = new StringBuffer();
    String[] keys = key.split(",");
    for (String k : keys) {
      if (k.trim().length() == 0) {
        continue;
      }
      /**
      * 根據(jù) dictCode 和 code 查詢字典值,例如:dictCode:sex,code:1,返回:男
      * 應(yīng)該放在redis,提高響應(yīng)速度
      */
      SysDictDetail dictData = sysDictService.getDictDataByTypeAndValue(dictType, key);
      if (dictData.getName() != null) {
        if (!"".equals(textValue.toString())) {
          textValue.append(",");
        }
        textValue.append(dictData.getName());
      }
      log.info("數(shù)據(jù)字典翻譯: 字典類型:{},當(dāng)前翻譯值:{},翻譯結(jié)果:{}", dictType, k.trim(), dictData.getName());
    }
    return textValue.toString();
  }
}

3、注解使用

我們只需要在實體類的屬性上加入我們實現(xiàn)的自定義注解即可

@ApiModelProperty(value = "性別(0--未知1--男2--女)")
@TableField("sex")
@Dict(dictCode = "sex")
private Integer sex;

@ApiModelProperty(value = "狀態(tài)(0--正常1--凍結(jié))")
@TableField("status")
@Dict(dictCode = "status")
private Integer status;

我們對 sex,status 都加入了 @Dict(dictCode = “”) 注解,那么我們在獲取用戶信息的時候,就能獲取到對應(yīng)的字典值了

五、測試

1、編寫 API 查詢

我們在 controller 層開放一個 API 實現(xiàn)查詢用戶列表

/**
 * 分頁查詢
 */
@PostMapping("list")
@ResponseBody
public LayTableResult list(Integer page, Integer limit, SysUserInfo userInfo) {
  QueryWrapper<SysUserInfo> queryWrapper = new QueryWrapper<>();
  if (StringUtils.isNotBlank(userInfo.getName())) {
    queryWrapper.like("name", userInfo.getName());
  }
  if (null != userInfo.getSex()) {
    queryWrapper.eq("sex", userInfo.getSex());
  }
  if (null != userInfo.getStatus()) {
    queryWrapper.eq("status", userInfo.getStatus());
  }
  queryWrapper.orderByDesc("create_time");
  IPage<SysUserInfo> iPage = sysUserInfoService.page(new Page<>(page, limit), queryWrapper);
  return new LayTableResult<>(iPage.getTotal(), iPage.getRecords());
}

注意: 這里我們使用了 LayTableResult 作為相應(yīng)實體類,與上面我們編寫的返回通用實體類是一致的,必須一直,才能實現(xiàn)數(shù)據(jù)字典功能

2、調(diào)用 API

返回結(jié)果如下:

{
	"code": 0,
	"msg": null,
	"count": 3,
	"data": [{
		"id": 2,
		"account": "15286779045",
		"name": "周杰倫",
		"sex": 1,
		"sexText": "男",
		"status": 0,
		"statusText": "正常"
	}, {
		"id": 1,
		"name": "超級管理員",
		"account": "15286779044",
		"sex": 1,
		"sexText": "男",
		"status": 0,
		"statusText": "正常"
	}]
}

可以看出,返回的數(shù)據(jù)中,多出了 sexText,statusText,兩個屬性,也就證明我們的字典功能已經(jīng)實現(xiàn)成功

六、總結(jié)

1、優(yōu)點(diǎn)

1、在一定程度上,通過系統(tǒng)維護(hù)人員即可改變系統(tǒng)的行為(功能),不需要開發(fā)人員的介入。使得系統(tǒng)的變化更快,能及時響應(yīng)客戶和市場的需求。
2、提高了系統(tǒng)的靈活性、通用性,減少了主體和屬性的耦合度 3、簡化了主體類的業(yè)務(wù)邏輯 4、
能減少對系統(tǒng)程序的改動,使數(shù)據(jù)庫、程序和頁面更穩(wěn)定。特別是數(shù)據(jù)量大的時候,能大幅減少開發(fā)工作量
5、使數(shù)據(jù)庫表結(jié)構(gòu)和程序結(jié)構(gòu)條理上更清楚,更容易理解,在可開發(fā)性、可擴(kuò)展性、可維護(hù)性、系統(tǒng)強(qiáng)壯性上都有優(yōu)勢。

2、缺點(diǎn)

1、數(shù)據(jù)字典是通用的設(shè)計,在系統(tǒng)效率上會低一些。
2、程序算法相對復(fù)雜一些。
3、對于開發(fā)人員,需要具備一定抽象思維能力,所以對開發(fā)人員的要求較高。

3、優(yōu)化

我們的數(shù)據(jù)字典數(shù)據(jù)應(yīng)該存放在 redis 中,減少與數(shù)據(jù)庫的交互次數(shù),提高響應(yīng)速度

到此這篇關(guān)于SpringBoot中實現(xiàn)數(shù)據(jù)字典的示例代碼的文章就介紹到這了,更多相關(guān)SpringBoot 數(shù)據(jù)字典內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java運(yùn)行shell腳本方法示例

    java運(yùn)行shell腳本方法示例

    利用Runtime.execute方法,我們可以在Java程序中運(yùn)行Linux的Shell腳本,或者執(zhí)行其他程序
    2013-12-12
  • SpringBoot web開發(fā)源碼深入分析

    SpringBoot web開發(fā)源碼深入分析

    Web開發(fā)的核心內(nèi)容主要包括內(nèi)嵌的Servlet容器和SpringMVCSpringBoot使用起來非常簡潔,大部分配置都有SpringBoot自動裝配,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)吧
    2022-10-10
  • JavaWeb入門教程之分頁查詢功能的簡單實現(xiàn)

    JavaWeb入門教程之分頁查詢功能的簡單實現(xiàn)

    這篇文章主要介紹了JavaWeb入門教程之分頁查詢功能的簡單實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-11-11
  • Java解碼H264格式視頻流中的圖片

    Java解碼H264格式視頻流中的圖片

    這篇文章主要為大家詳細(xì)介紹了Java解碼H264格式視頻流中的圖片,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-02-02
  • java封裝全局異常處理深入詳解

    java封裝全局異常處理深入詳解

    這篇文章主要為大家介紹了java封裝全局異常處理的深入詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • 使用maven插件對java工程進(jìn)行打包過程解析

    使用maven插件對java工程進(jìn)行打包過程解析

    這篇文章主要介紹了使用maven插件對java工程進(jìn)行打包過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-08-08
  • 解決Java包裝類比較時遇到的問題

    解決Java包裝類比較時遇到的問題

    所謂包裝類的作用就是將原始數(shù)據(jù)類型轉(zhuǎn)換成引用數(shù)據(jù)類型,下面這篇文章主要給大家介紹了關(guān)于在Java包裝類比較時遇到的問題的解決方法,文中給出了詳細(xì)的示例代碼,需要的朋友可以參考借鑒,下面來一起看看吧。
    2017-09-09
  • 淺談Java中的LinkedHashSet哈希鏈表

    淺談Java中的LinkedHashSet哈希鏈表

    這篇文章主要介紹了淺談Java中的LinkedHashSet哈希鏈表,LinkedHashSet 是 Java 中的一個集合類,它是 HashSet 的子類,并實現(xiàn)了 Set 接口,與 HashSet 不同的是,LinkedHashSet 保留了元素插入的順序,并且具有 HashSet 的快速查找特性,需要的朋友可以參考下
    2023-09-09
  • SpringBoot整合RocketMQ實現(xiàn)發(fā)送同步消息

    SpringBoot整合RocketMQ實現(xiàn)發(fā)送同步消息

    RocketMQ 是一款開源的分布式消息中間件,由阿里巴巴開源,它具有高可用性、高性能、低延遲等特點(diǎn),廣泛應(yīng)用于阿里巴巴集團(tuán)內(nèi)部以及眾多外部企業(yè)的業(yè)務(wù)系統(tǒng)中,本文給大家介紹了SpringBoot整合RocketMQ實現(xiàn)發(fā)送同步消息,需要的朋友可以參考下
    2024-04-04
  • SpringBoot調(diào)用Poi-tl實現(xiàn)渲染數(shù)據(jù)并生成Word文檔

    SpringBoot調(diào)用Poi-tl實現(xiàn)渲染數(shù)據(jù)并生成Word文檔

    這篇文章主要為大家詳細(xì)介紹了SpringBoot如何調(diào)用Poi-tl實現(xiàn)渲染數(shù)據(jù)并生成Word文檔,文中的示例代碼講解詳細(xì),有需要的小伙伴可以了解下
    2023-09-09

最新評論