对于一个比较大的业务系统,我们总是无止境的增加,删除,修改,粘贴,复制,想想总让人产生一种抗拒的心里。那有什么办法可以在正常的开发进度下自动生成一些类,配置文件,或者接口呢?
 

有感于马上要做个比较大的业务系统,想到那无止境的增删改查、粘贴复制,顿时脑后升起一阵凉风。于是想到个找或者写一个Java代码的生成器,这样在正常开发进度下,也能余下更多的时间去做些别的事情。

闲话少说,首先总结下需求:

我需要的这个工具能够读取数据库表结构,通过对字段类型、名称等分析得到需要的各种变量,根据模板生成相应的pojo类、hibernate的xml配置文件、dao和service的接口和类。

需求看起来很简单,但是一是没搞过这样的小工具,二是技术不过关,所以还是想到了找找有没有开源的代码拿过来根据自己的需求改。

于是找到了rapid-generator这个开源工具,同学们可以下载rapid-generator直接使用,根据自己的需求写好模板就行了。

由于自己项目中的特殊情况和公司规范等因素,另外也是想学习下别人的设计,所以我对源码进行了删减和功能的修改。

来看下主要的类:

Table:根据表结构建立的对象。

Column:根据表中每列建立的对象。

Generator:生成器核心类,主要负责根据表对象和读取FreeMarker模板生成最后的java代码文件。

GeneratorControl:控制生成过程等的一些参数,例如文件是否覆盖、文件编码等。

GeneratorProperties:读取配置文件的类,配置文件包括数据库连接信息和一些基本的参数配置。

下面来看一下模板的编写:

pojo模板:

  1. <#include "/java_copyright.include">
  2. <#assign className = table.className>
  3. <#assign classNameLower = className?uncap_first>
  4. package ${basepackage}.pojo.${mpackage}.${table.classNameFirstLower};
  5. <#include "/java_imports.include">
  6. import com.linkage.agri.pojo.base.BaseEntity;
  7. public class ${className} extends BaseEntity {
  8. private static final long serialVersionUID = 5454155825314635342L;
  9. <#list table.columns as column>
  10. /**
  11. * ${column.remarks}
  12. */
  13. private ${column.simpleJavaType} ${column.columnNameLower};
  14. </#list>
  15. <@generateJavaColumns/>
  16. <#macro generateJavaColumns>
  17. <#list table.columns as column>
  18. <#if column.isDateTimeColumn>
  19. public String get${column.columnName}String() {
  20. return DateConvertUtils.format(get${column.columnName}(), FORMAT_${column.constantName});
  21. }
  22. public void set${column.columnName}String(String ${column.columnNameLower}) {
  23. set${column.columnName}(DateConvertUtils.parse(${column.columnNameLower}, FORMAT_${column.constantName},${column.simpleJavaType}.class));
  24. }
  25. </#if>
  26. public void set${column.columnName}(${column.simpleJavaType} ${column.columnNameLower}) {
  27. this.${column.columnNameLower} = ${column.columnNameLower};
  28. }
  29. public ${column.simpleJavaType} get${column.columnName}() {
  30. return this.${column.columnNameLower};
  31. }
  32. </#list>
  33. </#macro>

FreeMarker的基本语法可以看下FreeMarker中文手册。

${}可以引用很多变量,这些变量包括:环境变量、table对象、配置变量等,这些变量都装在一个map里,如果自己有特殊需求,当然可以修改源码来装载更多的变量值。

注意:当变量为对象时,访问对象的属性是通过get方法来访问的。例如${table.classNameFirstLower},是引用 table对象的getNameFirstLower()方法,即使table对象中没有nameFirstLower这个属性也能够引用。

再看下我写的dao的模板:

  1. <#include "/java_copyright.include">
  2. <#assign className = table.className>
  3. <#assign classNameLower = className?uncap_first>
  4. package ${basepackage}.dao.${mpackage}.${table.classNameFirstLower};
  5. import java.math.BigDecimal;
  6. import java.util.List;
  7. import java.util.Map;
  8. import com.linkage.agri.dao.base.AbstractHibernateDAO;
  9. import com.linkage.agri.exception.DAOException;
  10. import ${basepackage}.pojo.${mpackage}.${classNameLower}.${className};
  11. <#include "/java_imports.include">
  12. public Class ${className}DAOImpl extends AbstractHibernateDAO implements ${className}DAO
  13. {
  14. /**
  15. * <query all>
  16. * @param paramMap
  17. * @param orderMap
  18. * @param pageNum
  19. * @param pageSize
  20. * @return
  21. * @throws DAOException
  22. */
  23. @SuppressWarnings
  24. public List<${className}> queryList${className}ByAttr(Map<String, Object> paramMap, Map<String, String> orderMap, int pageNum,
  25. int pageSize)
  26. throws DAOException
  27. {
  28. return super.listInstances(${className}.class, paramMap, orderMap, pageNum, pageSize);
  29. }
  30. /**
  31. *
  32. * <find one by id>
  33. * @param serial
  34. * @throws DAOException
  35. */
  36. public ${className} find${className}By${table.pkColumn.columnName}(${table.pkColumn.simpleJavaType} ${table.pkColumn.columnNameFirstLower})
  37. throws DAOException
  38. {
  39. return (${className})super.findById(${className}.class, ${table.pkColumn.columnNameFirstLower});
  40. }
  41. /**
  42. *
  43. * <save one>
  44. * @param ${table.classNameFirstLower}
  45. * @throws DAOException
  46. */
  47. public ${table.pkColumn.simpleJavaType} save${className}(${className} ${table.classNameFirstLower})
  48. throws DAOException
  49. {
  50. return (${table.pkColumn.simpleJavaType})super.saveInstance(${table.classNameFirstLower});
  51. }
  52. /**
  53. *
  54. * <update one>
  55. * @param ${table.classNameFirstLower}
  56. * @throws DAOException
  57. */
  58. public void update${className}(${className} ${table.classNameFirstLower})
  59. throws DAOException
  60. {
  61. super.updateInstance(${className});
  62. }
  63. /**
  64. * <check one is have?>
  65. * @param paramMap
  66. * @return
  67. * @throws DAOException
  68. */
  69. public boolean check${className}IsHaveByAttr(Map<String, Object> paramMap)
  70. throws DAOException
  71. {
  72. StringBuffer sqlBuffer = new StringBuffer();
  73. sqlBuffer.append("SELECT COUNT(*) FROM ${table.sqlName} T ");
  74. sqlBuffer.append("WHERE T.${table.pkColumn.sqlName} = ? ");
  75. BigDecimal big = (BigDecimal)super.findUniqueResultBySQLWithParams(sqlBuffer.toString(), paramMap.get("${table.pkColumn.columnNameFirstLower}"));
  76. return big.intValue() > 0 ? false : true;
  77. }
  78. /**
  79. * <update some>
  80. * @param ${table.classNameFirstLower}List
  81. * @return
  82. * @throws DAOException
  83. */
  84. public void update${className}Batch(List<${className}> ${table.classNameFirstLower}List)
  85. throws DAOException
  86. {
  87. super.updateBatchInstance(${table.classNameFirstLower}List);
  88. }
  89. /**
  90. *
  91. * <delete one>
  92. * @param ${table.classNameFirstLower}
  93. * @throws DAOException
  94. */
  95. public void delete${className}(${className} ${table.classNameFirstLower})
  96. throws DAOException
  97. {
  98. super.deleteInstance(${table.classNameFirstLower});
  99. }
  100. }

建立模板是能否解决自己问题的关键,在摸索出门道以后写模板变的十分简单。其实原理很简单,就是用一系列占位符来替换实际的变量值。

模板路径可以按照实际项目中的路径来拜访,生成器可以读取某个路径下所有模板进行生成,结果如下图:

接口和实现:

就说这么多吧,感兴趣的可以研究下这个工具的源码,也DIY出一套自己代码生成工具。不感兴趣的直接写模板就能用。

我只是走在路上,偶尔驻足观赏,偶尔回头看看。

告别无止境的增删改查:Java代码生成器的更多相关文章

  1. 告别无止境的增删改查--Java代码生成器

    转自:http://www.cnblogs.com/zhuYears/archive/2012/02/29/2373491.html 告别无止境的增删改查--Java代码生成器 有感于马上要做个比较大 ...

  2. 仿联想商城laravel实战---5、无刷新的增删改查(动态页面更新的三种方式(html))

    仿联想商城laravel实战---5.无刷新的增删改查(动态页面更新的三种方式(html)) 一.总结 一句话总结: 直接js增加删除修改html 控制器直接返回处理好的页面 用双向绑定插件比如vue ...

  3. salesforce 零基础学习(五十一)使用 Salesforce.com SOAP API 实现用户登录以及简单的增删改查(JAVA访问salesforce)

    此篇请参看:https://resources.docs.salesforce.com/202/latest/en-us/sfdc/pdf/salesforce_developer_environme ...

  4. MongoDB学习-->命令行增删改查&JAVA驱动操作Mongodb

    MongoDB 是一个基于分布式文件存储的数据库. 由 C++ 语言编写.旨在为 WEB 应用提供可扩展的高性能数据存储解决方案. MongoDB 是一个介于关系数据库和非关系数据库之间的产品,是非关 ...

  5. 5、Spring+Struts2+MyBatis+分页(mybatis无代理)增删改查

    1.创建如下项目结构 2.在src下的com.entity包下创建Dept.java package com.entity; /** * 部门表 * @author Holly老师 * */ publ ...

  6. HBase 增删改查Java API

    1. 创建NameSpaceAndTable package com.HbaseTest.hdfs; import java.io.IOException; import org.apache.had ...

  7. 增删改查-java(新手)

     PreparedStatement: 方法: Connection: 方法:  实例: 1.查询: package cn.chuang.JdbcDome; import java.sql.*; pu ...

  8. mysql数据库的连接以及增删改查Java代码实现(转载)

    每天叫醒自己的不是闹钟,而是梦想 数据库: create table t1(id int primary key not null auto_increment,name varchar(32),pa ...

  9. express无中间件的增删改查

    index.js const express = require("express");导入express框架 const data = require("./data& ...

随机推荐

  1. Oracle RAC Failover

    Oracle  RAC 同时具备HA(High Availiablity) 和LB(LoadBalance). 而其高可用性的基础就是Failover(故障转移). 它指集群中任何一个节点的故障都不会 ...

  2. 【Web学习日记】——C#引用WebService,从配置文件改变引用地址

    开发环境:Win7 32位,开发工具:VS2013,.Net:4.0 初用WebService,很多地方都搞不清楚怎么回事,但稍作研究之后,也就用上了,根本就没有考虑后续事情. 但是,随着项目的进行, ...

  3. LeapMotion(2):追踪五指

    上一篇文章,我们实现了Leap Motion的简单测试.追踪其中一个手指并用红色圆形表示其在空间的位置. 这篇文章,我们来实现五指的追踪. 其实,能够实现一指的追踪,那么五指的追踪自然不成问题.但是, ...

  4. 一个订单相关的存储过程(MySQL)

    BEGIN DECLARE currentDate VARCHAR(15) ;/*当前日期,有可能包含时分秒 */ DECLARE maxNo INT DEFAULT 0 ; /* 离现在最近的满足条 ...

  5. FastLoad错误 — RDBMS error 2634

    我们来看一下下面这条语句: BEGIN LOADING stu_flERRORFILES error_1, error_2;   如果此时已经存在error_1或error_2表,那么将会报错,信息如 ...

  6. 在Visual Studio 2010 中创建类库(dll)

    创建类库 选择"文件"->新建->项目->Visual C# ->类库,输入名称,选择位置,单击确定 浏览解决方案资源管理器,可以看到两个C#类,第一个是A ...

  7. 微软职位内部推荐-Senior Android Developer

    微软近期Open的职位: Position: Senior SDE-- Mobile Products Android/iOS/WP Senior Developer Contact Person: ...

  8. 简单制作mib表

    今天放假后第一天上班,将假前自学制作mib表的东西说一下. 在这里呢,我以世界-中国-上海-闵行这种包含关系介绍,感觉更容易理解. MIB file的开始和结束 所有的MIB file的都以DEFIN ...

  9. How to use Android Activity's finish(), onDestory() and System.exit(0) methods

    Activity.finish() Calling this method will let the system know that the programmer wants the current ...

  10. c++ std::string 用法

    std::string用法总结 在平常工作中经常用到了string类,本人记忆了不好用到了的时候经常要去查询.在网上摘抄一下总结一下,为以后的查询方便: string类的构造函数: string(co ...