news 2026/1/14 11:39:43

【然然管理系统】基于 SpringBoot+MyBatisPlus+Freemarker 实现代码生成功能(下)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【然然管理系统】基于 SpringBoot+MyBatisPlus+Freemarker 实现代码生成功能(下)

然然管理系统仓库地址,欢迎移步仓库点个小星星
https://gitee.com/OceanCore/ranran.git
https://github.com/qiaoting/ranran.git

一、前言

上篇我们梳理了代码生成功能的整体架构和核心模块,本篇将深入每个核心技术点,拆解关键代码的实现逻辑,同时分析开发中的踩坑点与优化方向,让你不仅能 “用”,还能 “懂” 并 “改造”。

系统截图:

二、核心点解析

1. 数据库表 / 字段查询:精准获取元数据

代码生成的前提是精准获取数据库元数据,我们基于information_schema实现,核心逻辑如下:

  • 表查询逻辑:通过GeneratorTableMapper.xmlselectAllTable方法,支持动态指定数据库名(未指定则用当前库),使用 MyBatis 的choose-when-otherwise实现条件分支:

    xml

    <choose> <when test="dbName != null and dbName != ''"> AND table_schema = #{dbName} </when> <otherwise> AND table_schema = DATABASE() </otherwise> </choose>
  • 字段查询逻辑:重点是字段属性的映射(比如isPrimaryKey/isNullable),通过case when将数据库的字符串结果(yes/no/pri)转为布尔值,适配 Java 实体类的属性类型。

踩坑点information_schema的字段名是区分大小写的(如column_name),不同 MySQL 版本可能有字段差异。

2. 数据库类型→Java 类型:精准映射

这是代码生成的核心环节之一,错误的类型映射会导致生成的代码无法编译。我们通过GenerateConstant定义类型常量,在GeneratorTableFieldService中实现映射逻辑:

步骤 1:定义类型常量(GenerateConstant)
public static final Set<String> STRING_TYPE = Set.of("char", "varchar", "varchar2", "nvarchar"); public static final Set<String> LONG_TYPE = Set.of("bit", "bigint", "integer"); public static final Set<String> INTEGER_TYPE = Set.of("int", "tinyint", "smallint", "mediumint"); public static final Set<String> DATE_TIME_TYPE = Set.of("datetime", "time", "date", "timestamp");
步骤 2:实现映射逻辑(GeneratorTableFieldService)
public List<GeneratorTableField> fetchTableFields(String tableName) { List<GeneratorTableField> generatorTableFields = generatorTableFieldMapper.selectAllField(tableName); for (GeneratorTableField field : generatorTableFields) { if (GenerateConstant.LONG_TYPE.contains(field.getColumnType())) { field.setJavaType("Long"); } else if (GenerateConstant.INTEGER_TYPE.contains(field.getColumnType())) { field.setJavaType("Integer"); } else if (GenerateConstant.DATE_TIME_TYPE.contains(field.getColumnType())) { field.setJavaType("LocalDateTime"); } else if (GenerateConstant.STRING_TYPE.contains(field.getColumnType())) { field.setJavaType("String"); } else { // 兜底:未匹配的类型默认BigDecimal(如decimal、double) field.setJavaType("BigDecimal"); } } return generatorTableFields; }

优化点:可扩展支持 JSON 类型(映射为JSONObject)、Blob 类型(映射为byte[]),或通过配置文件实现类型映射,避免硬编码。

3. 表名→Java 类名:优雅的驼峰转换

表名通常是下划线格式(如sys_user),而 Java 类名是大驼峰格式(如SysUser),TableUtil实现了这一转换,还支持移除指定前缀:

核心代码(TableUtil.convert)
public static String convert(String tableName, String[] removePrefixes) { if (!StrUtil.hasText(tableName)) { throw new IllegalArgumentException("表名不能为空"); } String lowerTableName = tableName.toLowerCase(); // 移除指定前缀(如t_、sys_) if (ObjUtil.isNotNull(removePrefixes)) { for (String prefix : removePrefixes) { if (lowerTableName.startsWith(prefix.toLowerCase())) { lowerTableName = lowerTableName.substring(prefix.length()); break; } } } // 下划线转大驼峰 String[] words = lowerTableName.split("_"); StringBuilder className = new StringBuilder(); for (String word : words) { if (!word.isEmpty()) { className.append(Character.toUpperCase(word.charAt(0))) .append(word.substring(1).toLowerCase()); } } if (className.isEmpty()) { throw new RuntimeException("转换后的类名不能为空,表名:" + tableName); } return className.toString(); }

关键设计

  • 先统一转小写,避免大小写不一致导致前缀匹配失败;
  • 支持自定义移除前缀(比如多系统表前缀不同);
  • 严格的参数校验,避免空表名 / 转换后空类名导致的异常。

踩坑点:表名全是下划线(如_test)会导致转换后类名为空,需在业务层提前校验。

4. Freemarker:模板渲染生成代码

Freemarker 是模板引擎,核心是 “数据模型 + 模板文件→生成字符串”,我们封装了FreemarkerService实现模板渲染:

步骤 1:Freemarker 初始化(PostConstruct)
@PostConstruct public void init() { configuration = new Configuration(); try { // 模板文件放在resources/ftl目录下 configuration.setClassForTemplateLoading(this.getClass(), "/ftl"); configuration.setDefaultEncoding("UTF-8"); // 解决中文乱码 configuration.setClassicCompatible(true); // 兼容空值渲染 configuration.setNumberFormat("0.##"); // 数字格式化 } catch (Exception e) { throw new RuntimeException("初始化freemarker配置失败", e); } }

踩坑点:模板路径配置错误(比如写成/templates/ftl)会导致找不到模板,需确认setClassForTemplateLoading的路径是resources下的相对路径。

步骤 2:模板渲染核心方法
public String generateString(String templateName, ClassInfoDto classInfoDto) { try { Template template = configuration.getTemplate(templateName); Writer writer = new StringWriter(); template.process(classInfoDto, writer); // 数据模型注入模板 return writer.toString(); } catch (Exception e) { throw new RuntimeException("freemarker模板渲染失败:" + templateName, e); } }
步骤 3:多模板渲染(GeneratorService)
public Map<String, String> generateCode(ClassInfoDto classInfoDto) { // 查询表字段并映射类型 List<GeneratorTableField> fields = generatorTableFieldService.fetchTableFields(classInfoDto.getTableName()); Map<String, String> codeMap = new HashMap<>(); // 组装类信息(模块名、类名、基础包名等) GenerateUtil.buildClassInfo(classInfoDto, fields); // 渲染不同模板,生成各类代码 codeMap.put("entity.java", generateEntity(classInfoDto)); codeMap.put("mapper.java", generateMapper(classInfoDto)); codeMap.put("service.java", generateService(classInfoDto)); codeMap.put("controller.java", generateController(classInfoDto)); return codeMap; }

核心设计ClassInfoDto作为统一的数据模型,包含表名、模块名、作者、字段列表、类名等所有模板所需的信息,实现 “一份数据模型,多模板复用”。

三、扩展与优化思路

基于当前实现,可从以下维度优化,让代码生成器更通用:

  1. 模板自定义:将模板文件从 jar 包内移到配置目录,支持用户上传自定义模板(如适配不同编码规范);
  2. 批量生成:支持前端选择多个表,批量生成代码并打包为 ZIP 返回;
  3. 类型映射配置化:将类型映射关系写入 YAML 配置文件,避免硬编码,示例:

    yaml

    generator: type-mapping: char: String varchar: String bigint: Long # 扩展JSON类型 json: JSONObject
  4. 前缀移除优化:当前TableUtil中移除前缀时break只移除第一个匹配的前缀,可改为循环移除所有匹配前缀(比如表名sys_t_user,移除sys_t_);
  5. 代码格式化:生成代码后通过JavaFormatter格式化代码,保证代码风格统一;
  6. 导入包自动处理:比如生成LocalDateTime时自动导入java.time.LocalDateTime,避免手动导包。

四、踩坑与解决方案

问题场景解决方案
Freemarker 渲染中文注释乱码初始化时设置configuration.setDefaultEncoding("UTF-8"),模板文件保存为 UTF-8 编码
表名转换后类名为空增加参数校验,对特殊表名(如全下划线)抛出明确异常
数据库类型映射遗漏兜底设置为BigDecimal,并在日志中打印未匹配的类型,便于后续补充
模板路径找不到确认setClassForTemplateLoading的路径是resources/ftl,且模板文件存在
五、总结与展望

本文深入解析了然然管理系统代码生成功能的核心细节,从数据库元数据查询、类型映射、表名转换到 Freemarker 模板渲染,每个环节都体现了 “职责单一、通用可扩展” 的设计思想。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/1/12 16:16:34

单目深度估计教程:MiDaS模型参数详解与调优

单目深度估计教程&#xff1a;MiDaS模型参数详解与调优 1. 引言&#xff1a;AI 单目深度估计的现实意义 在计算机视觉领域&#xff0c;从单张二维图像中恢复三维空间结构一直是极具挑战性的任务。传统方法依赖多视角几何或激光雷达等硬件设备&#xff0c;成本高且部署复杂。近…

作者头像 李华
网站建设 2026/1/12 16:15:44

基于SpringBoot的无人智慧超市管理系统毕业设计

博主介绍&#xff1a;✌ 专注于Java,python,✌关注✌私信我✌具体的问题&#xff0c;我会尽力帮助你。一、研究目的本研究旨在设计并实现一套基于SpringBoot框架的无人智慧超市管理系统&#xff0c;以应对现代零售业对智能化、高效化管理的需求。具体研究目的如下&#xff1a; …

作者头像 李华
网站建设 2026/1/12 16:14:13

如何高效解析长视频?Qwen3-VL-WEBUI一键部署,秒级定位关键事件

如何高效解析长视频&#xff1f;Qwen3-VL-WEBUI一键部署&#xff0c;秒级定位关键事件 在智能设备无处不在的今天&#xff0c;视频数据正以前所未有的速度积累——网课、会议录像、监控画面、直播回放。然而&#xff0c;“看得见”不等于“看得懂”。如何让AI真正理解一段两小…

作者头像 李华
网站建设 2026/1/12 16:06:48

零代码玩转AI分类:预置镜像5分钟上线,按分钟计费不浪费

零代码玩转AI分类&#xff1a;预置镜像5分钟上线&#xff0c;按分钟计费不浪费 1. 什么是智能分类&#xff1f;为什么电商需要它&#xff1f; 想象一下你刚接手一家线上服装店的运营工作&#xff0c;每天要上新数百件商品。手动给每件T恤打上"休闲""圆领"…

作者头像 李华
网站建设 2026/1/12 16:05:39

Eaton Tripp Lite 高速线缆的数据中心实测分析

在数据中心建设与升级过程中&#xff0c;高速线缆的性能稳定性直接影响网络吞吐、延迟和长期运行可靠性。Eaton 旗下的 Tripp Lite 作为全球较早进入数据中心基础设施领域的品牌&#xff0c;其高速线缆产品在服务器互联、交换机到机柜布线等场景中被大量采用。本文结合工程实践…

作者头像 李华
网站建设 2026/1/12 16:05:21

3步搞定AI分类:云端GPU免安装,小白也能玩转

3步搞定AI分类&#xff1a;云端GPU免安装&#xff0c;小白也能玩转 引言&#xff1a;告别复杂部署&#xff0c;AI分类从未如此简单 作为一名自媒体博主&#xff0c;你是否经常被海量的图片、视频素材搞得焦头烂额&#xff1f;每次找素材就像大海捞针&#xff0c;明明记得拍过…

作者头像 李华