这就是Java代码生成器的制作流程
云栖号资讯:【点击查看更多行业资讯】
在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来!
1. 前言
前几天写了篇关于Mybatis Plus代码生成器的文章,不少同学私下问我这个代码生成器是如何运作的,为什么要用到一些模板引擎,所以今天来说明下代码生成器的流程。
2. 代码生成器的使用场景
我们在编码中存在很多样板代码,格式较为固定,结构随着项目的迭代也比较稳定,而且数量巨大,这种代码写多了也没有什么技术含量,在这种情况下代码生成器可以有效提高我们的效率,其它情况并不适于使用代码生成器。
3. 代码生成器的制作流程
首先我们要制作模板,把样板代码的固定格式抽出来。然后把动态属性绑定到模板中,就像做填空题一样。所以在这个流程中模板引擎是最合适的。我们通过使用模板引擎的语法将数据动态地解析到静态模板中去,然后导出为编程中对应的文件就行了。
另外模板引擎有着丰富的绑定数据的指令集,可以让我们根据条件动态的绑定数据到模板中去。以Freemarker为例:
三元表达式:
${true ? 'checked': ''}
还有我们等下要用的遍历列表:
<#list fields as field> private ${field.fieldType} ${field.fieldName}; </#list>
在Java开发中我们常用的模板引擎有Freemarker、Velocity、Thymeleaf ,随着Web开发中前后端分离的流行模板引擎的使用场景正在被压缩,但是它依然是一门有用的技术。
4. 代码生成器演示
接下来,我们以Freemarker为例写一个简单的代码生成器,来生成POJO类。需要引入Freemarker的依赖。
<dependency> <groupId>org.freemarker</groupId> <artifactId>freemarker</artifactId> <version>2.3.28</version> </dependency>
4.1 模板制作
POJO的结构可以分为以下几部分:
java.lang 包无需导入。
所以将这些规则封装到配置类中:
public class JavaProperties { // 包名 private final String pkg; // 类名 private final String entityName; // 属性集合 需要改写 equals hash 保证名字可不重复 类型可重复 private final Set<Field> fields = new LinkedHashSet<>(); // 导入类的不重复集合 private final Set<String> imports = new LinkedHashSet<>(); public JavaProperties(String entityName, String pkg) { this.entityName = entityName; this.pkg = pkg; } public void addField(Class<?> type, String fieldName) { // 处理 java.lang final String pattern = "java.lang"; String fieldType = type.getName(); if (!fieldType.startsWith(pattern)) { // 处理导包 imports.add(fieldType); } Field field = new Field(); // 处理成员属性的格式 int i = fieldType.lastIndexOf("."); field.setFieldType(fieldType.substring(i + 1)); field.setFieldName(fieldName); fields.add(field); } public String getPkg() { return pkg; } public String getEntityName() { return entityName; } public Set<Field> getFields() { return fields; } public Set<String> getImports() { return imports; } /** * 成员属性封装对象. */ public static class Field { // 成员属性类型 private String fieldType; // 成员属性名称 private String fieldName; public String getFieldType() { return fieldType; } public void setFieldType(String fieldType) { this.fieldType = fieldType; } public String getFieldName() { return fieldName; } public void setFieldName(String fieldName) { this.fieldName = fieldName; } /** * 一个类的成员属性 一个名称只能出现一次 * 我们可以通过覆写equals hash 方法 然后放入Set * * @param o 另一个成员属性 * @return 比较结果 */ @Override public boolean equals(Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; Field field = (Field) o; return Objects.equals(fieldName, field.fieldName); } @Override public int hashCode() { return Objects.hash(fieldType, fieldName); } } }
接着就是静态模板entity.ftl
package ${pkg}; <#list imports as impt> import ${impt}; </#list> /** * the ${entityName} type * @author felord.cn */ public class ${entityName} { <#list fields as field> private ${field.fieldType} ${field.fieldName}; </#list> }
这里用到了Freemarker绑定数据的语法,比如List迭代渲染。
4.2 生成器编写
Freemarker通过声明配置并获取模板对象freemarker.template,该对象的process方法可以将动态数据绑定到模板中并导出为文件,最终实现了代码生成器,核心代码如下:
/** * 简单的代码生成器. * * @param rootPath maven 的 java 目录 * @param templatePath 模板存放的文件夹 * @param templateName 模板的名称 * @param javaProperties 需要渲染对象的封装 * @throws IOException the io exception * @throws TemplateException the template exception */ public static void autoCodingJavaEntity(String rootPath, String templatePath, String templateName, JavaProperties javaProperties) throws IOException, TemplateException { // freemarker 配置 Configuration configuration = new Configuration(Configuration.DEFAULT_INCOMPATIBLE_IMPROVEMENTS); configuration.setDefaultEncoding("UTF-8"); // 指定模板的路径 configuration.setDirectoryForTemplateLoading(new File(templatePath)); // 根据模板名称获取路径下的模板 Template template = configuration.getTemplate(templateName); // 处理路径问题 final String ext = ".java"; String javaName = javaProperties.getEntityName().concat(ext); String packageName = javaProperties.getPkg(); String out = rootPath.concat(Stream.of(packageName.split("\\.")) .collect(Collectors.joining("/", "/", "/" + javaName))); // 定义一个输出流来导出代码文件 OutputStreamWriter outputStreamWriter = new OutputStreamWriter(new FileOutputStream(out)); // freemarker 引擎将动态数据绑定的模板并导出为文件 template.process(javaProperties, outputStreamWriter); }
通过执行以下代码即可生成一个UserEntity的POJO:
// 路径根据自己项目的特点调整 String rootPath = "C:\\Users\\felord\\IdeaProjects\\codegenerator\\src\\main\\java"; String packageName = "cn.felord.code"; String templatePath = "C:\\Users\\felord\\IdeaProjects\\codegenerator\\src\\main\\resources\\templates"; String templateName = "entity.ftl"; JavaProperties userEntity = new JavaProperties("UserEntity", packageName); userEntity.addField(String.class, "username"); userEntity.addField(LocalDate.class, "birthday"); userEntity.addField(LocalDateTime.class, "addTime"); userEntity.addField(Integer.class, "gender"); userEntity.addField(Integer.class, "age"); autoCodingJavaEntity(rootPath, templatePath, templateName, userEntity);
生成的效果是不是跟手写的差不多:
5. 总结
这就是大部分代码生成器的机制,希望可以解答一些网友的疑问。
【云栖号在线课堂】每天都有产品技术专家分享!
课程地址:https://yqh.aliyun.com/live立即加入社群,与专家面对面,及时了解课程最新动态!
【云栖号在线课堂 社群】https://c.tb.cn/F3.Z8gvnK
原文发布时间:2020-07-15
本文作者:码农小胖哥
本文来自:“掘金”,了解相关信息可以关注“掘金”

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
有效提高java编程安全性的12条黄金法则
云栖号资讯:【点击查看更多行业资讯】在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来! 安全性是软件开发中最复杂,最广泛和最重要的考量之一。Java是具有许多内置安全性功能的开发平台,java在长期的发展过程中,已经经过了很多高强度的安全测试,并经常更新安全漏洞。并且Java生态系统还包括用于分析和报告安全性问题的各种工具。 但是,即使有了可靠的开发平台,也同样要保持警惕。应用程序开发是一项复杂的工作,漏洞会隐藏起来不易发现。程序员应该在应用程序开发的每个阶段都考虑安全性,从类级别的代码编程到API端点授权都应该被考虑在内。 以下为大家介绍的基本规则,可以为构建更安全的Java应用程序奠定良好的基础。供参考学习! Java安全规则1:编写简单而强大的Java代码 漏洞喜欢隐藏在复杂代码中,因此在不牺牲功能的情况下使代码尽可能简单。在代码中公开尽可能少的信息,隐藏实施细节,支持可维护和安全的代码。下面三个技巧将大大有助于编写安全的Java代码: 充分利用Java的访问修饰符。为类,方法及其属性声明访问级别,可以设为private的所有内容都应该为private。 避免过度使用...
- 下一篇
边缘计算在商业物联网部署中的作用
边缘计算能够分析和过滤原始数据集,并且仅将有价值的信息发送回云或数据中心。 几乎所有工业领域都已开始引入物联网(IoT)设备,预计将有58亿个终端投入使用。 在未来十年中,物联网设备预计将使移动和PC设备相形见绌,这意味着我们需要一种新型的网络解决方案来处理如此多的数据。 边缘计算被认为是一种进化,它将使计算和存储更接近终点。因此,电力智能电表或CCTV系统无需连续连接互联网即可运行。 通过在边缘处理传入的数据,更少的信息需要发送到云中并返回。这也大大减少了处理延迟。 ''Supermicro物联网副总裁MichaelClegg说道。“一个很好的类比是一家受欢迎的披萨餐厅,因为它在主要地点烘焙的馅饼在派往远方顾客的路上变冷,因此它在更多地区开设了较小的分支机构。” 根据Gartner的调查,到2023年,将有50%的大企业部署至少六个边缘计算用例,而在2019年,这一比例仅为1%。探索的大量增加和实施将导致收集的数据量激增。 幸运的是,边缘计算在这方面也有帮助,因为它能够分析和过滤原始数据集,并且仅将有价值的信息发送回数据中心。 在保持数据质量或范围的同时,这应该使网络成本比以前低。通...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2初体验,简单认识spring boot2并且搭建基础工程
- SpringBoot2全家桶,快速入门学习开发网站教程
- Docker安装Oracle12C,快速搭建Oracle学习环境
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- Eclipse初始化配置,告别卡顿、闪退、编译时间过长
- MySQL8.0.19开启GTID主从同步CentOS8
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- 设置Eclipse缩进为4个空格,增强代码规范