基于AOP思想的查询结果字典翻译-支持多级嵌套翻译

背景

之前在做的一个项目有一些列表查询的需求,查询结果某些字段需要进行翻译展示。例如省市编码需要转换为中文名称在页面列表上展示。当时比较急,然后需要转换的字段说多也不算很多,直接手撸了类似下面的代码出来
在这里插入图片描述
大概思路是将字段的翻译关系存在数据库一张表中,从es查出结果之后,再和从数据库查的翻译关系进行匹配翻译,然后在查数据库翻译关系的地方加了缓存。
用,肯定是能用的。不过现在再回看自己写的代码,不错,又堆出了一座屎山出来。。。
于是最近就在思考有没有什么优化的思路方法,然后就在网上找到了easy_trans和 JEECG。然后就参考这两款开源软件,结合自身需求写出了适合自己的easy_trans: gitee地址 使用该功能只需在数据库中配置好翻译关系字典表,在实体类中加上对应注解,即可在返回数据时对先关数据进行自动翻译,不需要在写任何多余代码,可以是代码看上去舒服很多。

参考

首先是easy_trans 官网地址
这是一款专注于数据翻译的开源软件,使用者需引入相关的maven依赖,做好先关配置后使用。其实这款开源软件从功能上来说是非常强大且符合我的背景需求的,我也是把官方提供的demo拉下来自己使用体验了一下。之所以没有选用有一下几点

  • 由于功能强大,感觉上很臃肿,所以需要做的配置等特别多,而我目前并不需要使用这么多的功能
  • 个人感觉对代码的侵入性有些强,特别是需要使用dictionaryTransService将字典缓存刷新到翻译服务
  • 再一个就是真的要小吐槽一下,官方提供的demo和文档看起来真的很费劲(不知道是我菜了还是大家也这么认为)

然后就是JEECG了,最后主要借鉴的也是JEECG的代码与思想。在从service层返回结果给ctrl时做一个AOP增强,对于注解了Dcit注解的实体类属性,在返回ctrl时会多返一个_dictText后缀的字段标识该属性的翻译值。

使用

代码引入

将下面红框中的代码复制到自己的项目中
其中redis配置如果自己项目已有可以使用自己项目原来的(不配置redis缓存也可以,会默认使用jvm内存做缓存)
com.thz.demo.model.result下的类可根据自己项目的返回结果进行相关调整,详细看后面的切面的parseDictText方法详解
在这里插入图片描述

字典表建立

字典表sql及本demo的相关测试sql已提供,根据自身需求执行
在这里插入图片描述

需要现在字典表中配置好翻译关系
label为翻译结果字段,value为翻译原值,type用以区分翻译那个字段
tree_node_type与father_id配合可支持父子嵌套的多级翻译
在这里插入图片描述

实体对象翻译注解添加

在需要翻译的字段上加上注解
dictType用于在字典表中筛选本字段的翻译数据
dictChildren用于表示依赖于自己的子字段有哪些
treeNodeType表示自己在树结构的第几次,不填写默认为根节点在第一层
在这里插入图片描述

切面调整

切点根据自己的需要,选择需要切入那些类下的哪些方法
在这里插入图片描述
判断方法的返回值类型后,取出相应结果传入handleRecords方法中进行解析增强
这一部分和切点一样需要根据自身项目情况修改
在这里插入图片描述

demo演示结果

demo中通提供了一个查询方法,可以看到是没有手撸任何翻译代码的
在这里插入图片描述
我们调用接口进行数据查询,可以看到返回结果中除了返回了数据库表数据,还一并返回了各个字段的翻译值
在这里插入图片描述

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在 Spring AOP 中实现返回前字典翻译,可以使用自定义注解和 AOP 切面来实现。 首先,在返回值需要翻译的方法上添加自定义注解,例如 @DictTranslate: ```java @DictTranslate public List<User> getUsers() { // ... } ``` 然后,编写 AOP 切面,在方法执行完成后,对返回值进行翻译操作。可以使用 Map 存储字典值和翻译后的值的对应关系,使用反射获取方法返回值类型,判断是否是需要翻译的类型,然后遍历返回值进行翻译操作。 ```java @Aspect @Component public class DictTranslateAspect { // 字典值和翻译后的值的对应关系 private static final Map<String, String> DICT_MAP = new HashMap<>(); // 切入点,对添加了 @DictTranslate 注解的方法进行切面操作 @Pointcut("@annotation(com.example.DictTranslate)") public void dictTranslate() {} // 方法执行完成后,对返回值进行翻译操作 @AfterReturning(pointcut = "dictTranslate()", returning = "returnValue") public Object doAfterReturning(JoinPoint joinPoint, Object returnValue) throws Throwable { MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature(); Class<?> returnType = methodSignature.getReturnType(); // 判断返回值类型是否是需要翻译的类型 if (List.class.isAssignableFrom(returnType)) { List<?> list = (List<?>) returnValue; // 遍历返回值进行翻译操作 for (Object obj : list) { Field[] fields = obj.getClass().getDeclaredFields(); for (Field field : fields) { if (field.isAnnotationPresent(Dict.class)) { field.setAccessible(true); Object value = field.get(obj); String dictType = field.getAnnotation(Dict.class).value(); String dictValue = DICT_MAP.get(dictType + "_" + value); if (dictValue != null) { field.set(obj, dictValue); } } } } } return returnValue; } } ``` 在上述代码中,@DictTranslate 注解用于标记需要进行翻译的方法,DictTranslateAspect 类是 AOP 切面,用于在方法执行完成后对返回值进行翻译操作。@Pointcut 注解用于定义切入点,@AfterReturning 注解用于定义方法执行完成后的操作。 需要注意的是,@Dict 注解用于标记需要翻译的字段,其 value 属性表示字典的类型。DICT_MAP 变量是字典值和翻译后的值的对应关系,可以从数据库或其他数据源中加载字典数据。在翻译操作中,使用 field.getAnnotation(Dict.class).value() 获取字典类型,使用 field.get(obj) 获取需要翻译的值,然后在 DICT_MAP 中查找对应的翻译后的值,最后使用 field.set(obj, dictValue) 将翻译后的值设置回对象中。 这样,在方法返回前,就可以完成对字典值的翻译操作了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值