基于mybatis-plus的多语言扩展

概览

对于表中字段,需要实现多语言的方案探讨:

1.表中横向扩展多个字段分别存储中文,英文,俄语等语言字段,查询时,根据需要查询的语言,进行查询

2.增加一张多语言表,存储多语言信息,如图,同时基于mybatis-plus扩展插入更新查询等功能,完成代码无侵入的多语言功能


一、组件的开发

1.自定义重写sql注入器ZlxSqlInjector

  @Overridepublic void inspectInject(MapperBuilderAssistant builderAssistant, Class<?> mapperClass) {Class<?> modelClass = ReflectionKit.getSuperClassGenericType(mapperClass, Mapper.class, 0);if (modelClass != null) {String className = mapperClass.toString();Set<String> mapperRegistryCache = GlobalConfigUtils.getMapperRegistryCache(builderAssistant.getConfiguration());if (!mapperRegistryCache.contains(className)) {List<AbstractMethod> methodList = this.getMethodList(mapperClass);if (CollectionUtils.isNotEmpty(methodList)) {ZlxTableInfo tableInfo = ZlxTableInfoHelper.initTableInfo(builderAssistant, modelClass,className);// 循环注入自定义方法methodList.forEach(m -> m.inject(builderAssistant, mapperClass, modelClass, tableInfo));} else {log.debug(mapperClass.toString() + ", No effective injection method was found.");}mapperRegistryCache.add(className);}}}@Overridepublic List<AbstractMethod> getMethodList(Class<?> mapperClass) {return Stream.of(new com.zenglx.multilanguage.method.Insert(),new Delete(),new DeleteByMap(),new DeleteById(),new DeleteBatchByIds(),new Update(),new UpdateById(),new com.zenglx.multilanguage.method.SelectById(),new com.zenglx.multilanguage.method.SelectBatchByIds(),new SelectByMap(),new com.zenglx.multilanguage.method.SelectOne(),new com.zenglx.multilanguage.method.SelectCount(),new SelectMaps(),new SelectMapsPage(),new SelectObjs(),new com.zenglx.multilanguage.method.SelectList(),new com.zenglx.multilanguage.method.SelectPage()).collect(toList());}

2.自定义元数据和方法

@Getter
@Setter
public class ZlxTableInfo extends TableInfo {private String condition = SqlCondition.EQUAL;/*** 多语言字段*/private Set<TableFieldInfo> multiLanguageColumns = new HashSet<>();/*** 实体类 => 主键信息*/private Set<TableFieldInfo> entityClassPkColumns = new HashSet<>();/*** 实体类 => 全部列*/private Set<TableFieldInfo> entityClassColumns = new HashSet<>();/*** 是否多语言*/private boolean multiLanguage;public ZlxTableInfo(Class<?> entityType) {super(entityType);}public List<String> multiLanguageColumns() {return multiLanguageColumns.stream().map(TableFieldInfo::getColumn).collect(Collectors.toList());}/*** 获取所有的查询的 sql 片段** @param ignoreLogicDelFiled 是否过滤掉逻辑删除字段* @param withId              是否包含 id 项* @param prefix              前缀* @return sql 脚本片段*/@Overridepublic String getAllSqlWhere(boolean ignoreLogicDelFiled, boolean withId, final String prefix) {final String newPrefix = prefix == null ? EMPTY : prefix;String filedSqlScript = super.getFieldList().stream().filter(i -> {if (ignoreLogicDelFiled) {return !(isWithLogicDelete() && i.isLogicDelete());}return true;}).map(i -> getSqlWhere(newPrefix,i)).filter(Objects::nonNull).collect(joining(NEWLINE));if (!withId || StringUtils.isBlank(super.getKeyProperty())) {return filedSqlScript;}String newKeyProperty = newPrefix + super.getKeyProperty();String keySqlScript = super.getKeyColumn() + EQUALS + SqlScriptUtils.safeParam(newKeyProperty);return SqlScriptUtils.convertIf(keySqlScript, String.format("%s != null", newKeyProperty), false)+ NEWLINE + filedSqlScript;}private String convertIfProperty(String prefix, String property) {return StringUtils.isNotBlank(prefix) ? prefix.substring(0, prefix.length() - 1) + "['" + property + "']" : property;}/*** 获取 查询的 sql 片段** @param prefix 前缀* @return sql 脚本片段*/public String getSqlWhere(final String prefix,TableFieldInfo tableFieldInfo) {final String newPrefix = prefix == null ? EMPTY : prefix;// 默认:  AND column=#{prefix + el}String sqlScript = " AND " + String.format(condition, tableFieldInfo.getColumn(), newPrefix + tableFieldInfo.getEl());if(this.getMultiLanguageColumns().contains(tableFieldInfo)) {sqlScript = " AND " + String.format(condition, "t."+tableFieldInfo.getColumn(), newPrefix + tableFieldInfo.getEl());}// 查询的时候只判非空return convertIf(sqlScript, convertIfProperty(newPrefix, convertIfProperty(newPrefix, tableFieldInfo.getProperty())), tableFieldInfo);}/*** 转换成 if 标签的脚本片段** @param sqlScript     sql 脚本片段* @param property      字段名* @param tableFieldInfo 验证策略* @return if 脚本片段*/private String convertIf(final String sqlScript, final String property, TableFieldInfo tableFieldInfo) {if (tableFieldInfo.getWhereStrategy() == FieldStrategy.NEVER) {return null;}if (tableFieldInfo.isPrimitive() || tableFieldInfo.getWhereStrategy() == FieldStrategy.IGNORED) {return sqlScript;}if (tableFieldInfo.getWhereStrategy() == FieldStrategy.NOT_EMPTY && tableFieldInfo.isCharSequence()) {return SqlScriptUtils.convertIf(sqlScript, String.format("%s != null and %s != ''", property, property),false);}return SqlScriptUtils.convertIf(sqlScript, String.format("%s != null", property), false);}}
public class SelectById extends com.baomidou.mybatisplus.core.injector.methods.SelectById {@Overridepublic MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) {ZlxTableInfo zlxTableInfo = (ZlxTableInfo)tableInfo;if(zlxTableInfo.isMultiLanguage()) {SqlMethod sqlMethod = SqlMethod.SELECT_BY_ID;String scriptSql = "<script>SELECT %s FROM %s WHERE %s \n</script>";String sql = String.format(scriptSql, SqlHelper.getAllColumns(tableInfo), SqlHelper.selectFromTableTl(tableInfo),sqlWhereEntityWrapper(true, tableInfo));SqlSource sqlSource = languageDriver.createSqlSource(configuration, sql, modelClass);return this.addSelectMappedStatementForTable(mapperClass, getMethod(sqlMethod), sqlSource, tableInfo);} else {return super.injectMappedStatement(mapperClass, modelClass, tableInfo);}}@Overrideprotected String sqlWhereEntityWrapper(boolean newLine, TableInfo tableInfo) {ZlxTableInfo zlxTableInfo = (ZlxTableInfo)tableInfo;if(zlxTableInfo.isMultiLanguage()) {StringBuilder sb = new StringBuilder();sb.append("b.").append(zlxTableInfo.getKeyColumn()).append("=#{id}");return sb.toString();} else {return super.sqlWhereEntityWrapper(newLine, tableInfo);}}
}

3.自定义拦截器

 public class ZlxMultiLanguageInterceptor extends PaginationInnerInterceptor {@Overridepublic void beforeUpdate(Executor executor, MappedStatement statement, Object parameter) throws SQLException {ZlxTableInfo tableInfo = ZlxTableInfoHelper.getTableByMapper(getMapperClassName(statement.getId()));if (tableInfo != null && tableInfo.isMultiLanguage()) {Connection connection = executor.getTransaction().getConnection();switch (statement.getSqlCommandType()) {case INSERT:insertMultiLanguage(tableInfo, parameter, connection);break;case UPDATE:List<String> columnNameList = tableInfo.getEntityClassColumns().stream().map(TableFieldInfo::getColumn).collect(Collectors.toList());updateMultiLanguage(tableInfo, parameter, connection, new HashSet<>(columnNameList));break;case DELETE:proceedDeleteMultiLanguage(tableInfo, parameter, connection);break;default:break;}}}}

二、组件的使用

1.新建表

原表t_user,多语言表t_user_tl

2.引入库

<dependency><groupId>com.zenglx.multilanguage</groupId><artifactId>multilanguage-spring-boot-starter</artifactId><version>1.0-SNAPSHOT</version>
</dependency>

3.demo验证测试

 

@RequestMapping("/v1/demo")
public class DemoController {@Autowiredprivate UserService userService;/*** 新增用户** @return*/@PostMappingpublic ResponseEntity add(@RequestBody User user) {userService.save(user);return ResponseEntity.ok("成功");}/*** 用户详情** @return*/@GetMappingpublic ResponseEntity<User> get(@RequestParam("id") Long id) {return ResponseEntity.ok(userService.getById(id));}/*** 查询用户列表** @return*/@GetMapping("/list")public ResponseEntity<List<User>> list() {return ResponseEntity.ok(userService.list());}/*** 更新用户** @return*/@PutMappingpublic ResponseEntity update(@RequestBody User user) {userService.update(user,null);return ResponseEntity.ok("成功");}/*** 删除用户** @return*/@DeleteMappingpublic ResponseEntity delete(@RequestParam("id") Long id) {userService.removeById(id);return ResponseEntity.ok("成功");}@GetMapping("/page")public ResponseEntity<Page<User>> page(@RequestParam(required = false, value = "name") String name,@RequestParam(defaultValue = "1", value = "page") Integer page,@RequestParam(defaultValue = "10", value = "pageSize") Integer pageSize) {Page<User> param = new Page<>(page,pageSize);LambdaQueryWrapper<User> lambdaQueryWrapper = new LambdaQueryWrapper<>();lambdaQueryWrapper.like(StringUtils.hasText(name),User::getName,name);Page<User> userPage = userService.page(param,lambdaQueryWrapper);return ResponseEntity.ok(userPage);}

   a.用户添加

b.用户查询


总结

可运行源码

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/17363.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

IC开发——VCS基本用法

1. 简介 VCS是编译型verilog仿真器&#xff0c;处理verilog的源码过程如下&#xff1a; VCS先将verilog/systemverilog文件转化为C文件&#xff0c;在linux下编译链接生成可执行文件&#xff0c;在linux下运行simv即可得到仿真结果。 VCS使用步骤&#xff0c;先编译verilog源…

深入理解Java中的ThreadLocal机制:原理、方法与使用场景解析

深入理解Java中的ThreadLocal机制&#xff1a;原理、方法与使用场景解析 ThreadLocal 是 Java 中提供的一种机制&#xff0c;用于在多线程环境下为每个线程提供独立变量副本&#xff0c;避免了线程间共享变量引发的并发问题。通过 ThreadLocal&#xff0c;每个线程都可以拥有自…

js面试题----3

61.new的原理是什么?通过new的方式创建对象和通过字面量创建有什么区别? new: 创建一个新对象。这个新对象会被执行[[原型]]连接。将构造函数的作用域赋值给新对象,即this指向这个新对象.如果函数没有返回其他对象,那么new表达式中的函数调用会自动返回这个新对象。functio…

Nuxt2 控制路由导航方式

控制路由导航 1 在插件中使用 beforeEach 守卫 首先&#xff0c;创建一个插件文件&#xff0c;例如 router.js&#xff1a; export default ({ app, store }) > {app.router.beforeEach((to, from, next) > {if (to?.fullPath /buy/ && from?.fullPath /p…

STM32--ADC

一、简介 *ADC&#xff08;Analog-Digital Converter&#xff09;模拟-数字转换器 *ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量&#xff0c;建立模拟电路到数字电路的桥梁 *12位逐次逼近型ADC&#xff0c;1us转换时间 *输入电压范围&#xff1a;0~3.3V&…

cv2.imdecode 和 cv2.imread 的区别

cv2.imdecode 和 cv2.imread 都是 OpenCV 用于读取图像的函数&#xff0c;但它们用于不同的场景&#xff0c;处理方式也不同。 cv2.imread 用法&#xff1a; img cv2.imread(image_path)功能&#xff1a; cv2.imread 用于直接从文件系统中读取图像文件。image_path 是图像文件…

做人脸识别系统,使用python什么个库?

在构建人脸识别系统时&#xff0c;选择合适的库至关重要。下面列出几种流行且功能强大的库&#xff0c;并根据不同需求和场景推荐使用&#xff1a; 1. FaceNet FaceNet 是谷歌开发的高性能人脸识别模型&#xff0c;具有很高的准确性和广泛的应用。 优点&#xff1a; 高精度的…

Flutter 中的 FractionallySizedBox 小部件:全面指南

Flutter 中的 FractionallySizedBox 小部件&#xff1a;全面指南 Flutter 的布局系统非常灵活&#xff0c;允许开发者以各种方式对组件进行尺寸调整。FractionallySizedBox 是 Flutter 中一个非常有用的布局小部件&#xff0c;它允许子组件的尺寸基于父组件的尺寸来计算。本文…

【备忘】Adobe XD提供了一系列的快捷键来帮助设计师们提高工作效率。以下是一些常用的Adobe XD快捷键及其用途:

Adobe XD提供了一系列的快捷键来帮助设计师们提高工作效率。以下是一些常用的Adobe XD快捷键及其用途&#xff1a;### Adobe XD 快捷键列表&#xff1a;#### 基本选择与编辑 - **V** - 选择工具&#xff0c;用于选择和移动元素。 - **A** - 直接选择工具&#xff0c;用于选择和…

redisson 释放分布式锁 踩坑

java.lang.IllegalMonitorStateException: attempt to unlock lock, not locked by current thread by node id: 48c213c9-1945-4c1b-821e-6d32e347eb44 thread-id: 69 出错代码&#xff1a; private void insertHourLog(Timestamp lastHourStartTimeStamp) {RLock lock red…

2024上半年软考高级系统架构设计师回顾

本博客地址&#xff1a;https://security.blog.csdn.net/article/details/139238685 2024年上半年软考在5月25-26日举行&#xff0c;趁着时间刚过去记忆还在&#xff0c;简单写一点总结。 关于考试形式&#xff1a;上机考试&#xff08;以后也都是机考&#xff09;&#xff0…

leetcode 1241每个帖子的评论数(postgresql)

需求 编写 SQL 语句以查找每个帖子的评论数。 结果表应包含帖子的 post_id 和对应的评论数 number_of_comments 并且按 post_id 升序排列。 Submissions 可能包含重复的评论。您应该计算每个帖子的唯一评论数。 Submissions 可能包含重复的帖子。您应该将它们视为一个帖子。…

BI工具如何为金融行业带来变革?金融行业营销管理策略大揭秘

当今数字化时代&#xff0c;金融行业正经历着前所未有的变革。随着大数据、人工智能、区块链等新兴技术的兴起&#xff0c;金融机构正面临着重新定义服务模式、风险管理和客户体验的挑战。商业智能&#xff08;BI&#xff09;作为这一变革的关键驱动力&#xff0c;已经成为金融…

ComfyUI工作流网站

https://openart.ai/home https://comfyworkflows.com/ https://civitai.com/

claude3国内API接口对接

众所周知&#xff0c;由于地理位置原因&#xff0c;Claude3不对国内开放&#xff0c;而国内的镜像网站使用又贵的离谱&#xff01; 因此&#xff0c;团队萌生了一个想法&#xff1a;为什么不创建一个一站式的平台&#xff0c;让用户能够通过单一的接口与多个模型交流呢&#x…

视频营销的智能剪辑:Kompas.ai如何塑造影响力视频内容

引言&#xff1a; 在当今数字化的营销领域&#xff0c;视频内容已经成为品牌吸引用户注意力、建立品牌形象和提升用户参与度的重要方式。然而&#xff0c;要想制作出具有影响力的视频内容&#xff0c;并不是一件容易的事情。这就需要借助先进的技术和工具&#xff0c;如人工智能…

《中国统计》文本分析技术最新进展总结盘点

《中国统计》文章展示了文本分析在各个领域的应用&#xff0c;包括大语言模型、科技文献、数据分类、乡村振兴、数据资产评估、历史文献解读、学科融合、基础设施管理和社情民意调查等&#xff0c;凸显了文本分析在数据挖掘和决策支持中的重要作用。 房祥忠.大语言模型中的统计…

FlatBuffers C++ 写一个例子

FlatBuffers 是一个用于序列化和反序列化数据的开源库&#xff0c;它强调高性能和灵活性。下面是一个简单的 FlatBuffers C 示例&#xff0c;该示例将展示如何定义一个简单的 FlatBuffers schema、生成 C 代码&#xff0c;并使用生成的代码来序列化和反序列化数据。 定义 Flat…

2024开放式蓝牙耳机推荐,五款性价比最高的耳机推荐

在我们的日常生活中&#xff0c;无论是上下班通勤还是锻炼身体&#xff0c;耳机都是我们放松心情、驱散无聊的好伙伴。不过&#xff0c;面对市场上不断涌现的开放式蓝牙耳机&#xff0c;挑选一款既符合个人喜好又满足需求的产品&#xff0c;确实需要一些技巧。今天&#xff0c;…

springboot实现多开发环境匹配置

首先logbok-spring.xml里面的内容 <?xml version"1.0" encoding"UTF-8"?> <configuration><!-- 开发、测试环境 --><springProfile name"dev,test"><include resource"org/springframework/boot/logging/log…