克隆/下载
贡献代码
同步代码
取消
提示: 由于 Git 不支持空文件夾,创建文件夹后会生成空的 .keep 文件
Loading...
README
Apache-2.0

1、什么是codgen?

Codgen是一个基于数据库元数据模型,使用freemarker模板引擎来构建输出的代码生成器。freemarker的数据模型结构通常来说都是一个Map树状结构模型,codgen也不例外,它的数据模型这棵树的根节点一般包含一个TableModel(表数据模型)对象。TableModel顾名思义就是由一张数据库表的元数据信息组成的一个数据模型,有了这个数据模型,再加上一套使用freemarker编写的多层架构模板,就可以生成一套基于这个表的多层架构代码文件。

总的来说,codgen具有以下主要功能及特性: 采用freemarker的构建公式:数据模型+模板=输出,默认使用freeMarker模板引擎来生成代码,但可以通过Builder接口实现其他构建方式。 核心数据模型TableModel基于JDBC实现表元数据及其所有字段列的相关元数据信息的封装。 数据模型TableModel基于JDBC实现,可以取得大部分元数据信息,个别信息的取得与具体数据库方言有关,可以通过扩展DbProvider?来实现。 通过实现接口ColumnHandler,可以完成JDBC数据类型到各种编程语言的类型转换操作及更多复杂的操作,如处理Oracle的大写列名以增强列名称的可读性。 通过配置可以动态增加或重定义数据模型,并可以被后面的数据模型通过模板语言引用或组装。 构建时指定的模板可以是一段文本字符串,也可以是一个文件路径,并且它们的内容里都可以引用已定义的数据模型。 构建时指定的输出类型可以是文本,也可以是文件,指定的文件输出路径也可以引用数据模型变量。 项目配置引入继承机制,这样就可以重用在父类配置中已定义的数据模型及其他配置信息。 项目配置信息可以分开多个配置文件存放,codgen一次性加载并缓存起来以加速之后的访问。

2、如何使用codgen

2.1 下载最新的codgen.jar及其依赖包

由于codgen不是一个独立的项目,本着不重新发明轮子的原则,依赖于一些比较流行的开源框架。因此除了需要下载最新的codgen.jar包文件以外,还要下载以下jar文件:

commons-lang.jar 或以上兼容版本 commons-logging.jar或以上兼容版本 log4j-1.2.14或以上兼容版本 freemarker-2.3.13或以上兼容版本 除了以上4个必须依赖的开源包以外,由于codgen的TableModel?元数据是通过JDBC来获取的,还需要下载对应数据库方言的JDBC驱动包,比如SQLServer2005的JDBC驱动包是sqljdbc.jar。目前codgen默认只支持Oracle10g\SQLServer2000\SQLServer2005三种数据库版本。但是可以通过扩展实现codgen提供的DbProvider?接口来支持更多的数据库版本。 Oracle10g的JDBC驱动包:ojdbc14.jar SQLServer2000的JDBC驱动包:msbase.jar、mssqlserver.jar和msutil.jar SQLServer2005的JDBC驱动包:sqljdbc.jar 注:JDK1.6.0_13或以上兼容版本

2.2 编写codgen配置文件

虽然不使用配置文件的方式也可以实现整个代码的构建操作,但是为了减少使用代码的编写量及方便日后维护,建议使用配置文件来声明构建所依赖的一些信息,譬如‘数据模型’变量声明,‘输出模型’变量声明,以及数据库信息提供者等一些可以与代码分离的配置信息。以下就是一段比较完整的配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE codgen-config PUBLIC "-//Apache Software Foundation//DTD Tengen Configuration 1.0//EN" "com/bcs/codgen/resources/codgen-config_1_0.dtd">
<codgen-config>
        <project name="AOWork" label="工程验收" outputEncoding="GBK" isDefault="true" extends="defaultProject">
                <dbProvider class="com.bcs.codgen.service.impl.Sql2005Provider">
                        <jdbcConfig>
                                <driver>com.microsoft.sqlserver.jdbc.SQLServerDriver</driver>
                                <url>jdbc:sqlserver://127.0.0.1\SQL2005;DatabaseName=MUCM_DG;selectMethod=cursor</url>
                                <user>sa</user>
                                <password>sa</password>
                        </jdbcConfig>
                        <columnHandler class="com.bcs.codgen.service.impl.DataTypeConverterForJava"/>
                </dbProvider>
                
                <dataModel name="templateDirectory">template/AOWork</dataModel>
                <dataModel name="outputDirectory">D:/QtoneProject/TengenCode/${copyright.author}/${projectName}</dataModel>
                <dataModel name="NamespaceModel" >com.qtone.aow.model.${groupName}</dataModel>
                <dataModel name="NamespaceParam">com.qtone.aow.model.${groupName}</dataModel>
                <dataModel name="NamespaceDao">com.qtone.aow.dao.${groupName}</dataModel>
                <dataModel name="NamespaceDaoImpl">com.qtone.aow.dao.${groupName}</dataModel>
                <dataModel name="NamespaceBll">com.qtone.aow.bll.${groupName}</dataModel>
                <dataModel name="NamespaceBllImpl">com.qtone.aow.bll.${groupName}</dataModel>
                <dataModel name="NamespaceApp">com.qtone.aow.app.${groupName}</dataModel>
                                
                <template name="tDao" type="file">dao.ftl</template>
                
                <output name="Model" type="text" templateText="${defaultJavaModel}" ></output>
                <output name="Param" type="text" templateFile="Param.ftl" ></output>
                <output name="Dao" type="text" templateName="tDao"></output>
                <output name="DaoImpl" type="text"></output>
                <output name="Bll" type="file" templateFile="template/AOWork/Bll">${outputDirectory}/Bll.java</output>
                <output name="BllImpl" type="file">${outputDirectory}/BllImpl.java</output>
        </project>
</codgen-config>

2.2.1 project节点

一个项目就是一个最小构建单位。

  • name 项目名称,在加载的所有配置中必须保证项目名称的唯一性。
  • label 项目标签,中文描述信息,一般用作前台显示标签。
  • outputEncoding 输出编码类型,必须指定与模板的编码类型一致,否则输出的文件或会出现乱码。
  • isDefault 是否默认项目,如果有多个项目设置为默认属性(isDefault="true"),则按加载的先后顺序以最后一个为准。
  • extends 指定继承一个之前已配置的现有项目,在加载并创建一个项目配置对象时,如果发现设置了该属性,就会先取得被继承的一个项目配置副本,在这个基础上再读取当前项目的配置信息,如果存在名称相同的子节点,就会覆盖。
  • isEnabled 是否启用,如果不设置该属性,则默认为启用状态,相当于设置了isEnabled ="true",已禁用的项目不会被加载到配置缓存里,因此也不能被之后的项目所继承。

2.2.2 dbProvider节点

该节点用于配置当前项目所使用的数据库信息提供者,非强制性配置节点。dbProvider一般包含一个jdbcConfig和一个或多个columnHandler等子节点,但是这些也不是强制需要配置的,可以在编程里面再指定,譬如jdbcConfig在不配置的情况下,可以在编程的时候在指定一个JdbcConfig对象或者直接传递一个数据库连接Connection对象即可。也就是说DbProvider分别提供了JdbcConfig和Connection作为参数的两种构造方法。DbProvider配置节点只有一个class属性,该属性指定的就是具体某种数据库方言的一个数据库元数据信息提供者的一个实现,譬如OracleProvider就是一个DbProvider的实现类,用于提供Oracle数据库相关的一些元数据信息。codgen还实现了MsSQL等一些数据库的DbProvider,当然,你也可以通过继承DbProvider来扩展自己的数据库信息提供者,这样的工作也很简单,毕竟DbProvider抽象类已经实现了大部分的功能,你只需要实现一两个抽象方法就可以扩展出一个自己的dbProvider,在配置里面使用class属性指定该类的完全限定名称即可。

2.2.3 jdbcConfig节点

该节点用于指定JDBC连接的配置信息,包含driver、url、user和password等子节点。

2.2.4 columnHandler节点

columnHandler指的是列模型处理器。它是dbProvider下的子节点,一个数据库信息提供者可以包含多个列模型处理器,在创建表模型时从数据库里取得相关字段列信息并封装成一个列模型对象后,这些处理器就会被调用来进行后续的一些特殊处理操作,比如根据把数据库字段的数据类型转换成实际编程语言的数据类型,再如Oracle的所有字段名称都是大写的,在这里可以增加一个处理器负责把这些大写的名称统一转换成可读性比较强的列名称,如下面代码所示: ProjectConfig projectConfig = ProjectConfigHelper.getDefaultProjectConfig(null);

//增加一个额外的列模型处理器,处理Oracle的大写列名以增强列名称的可读性

projectConfig.getDbProvider().getColumnHandlers().add(new ColumnHandler() {
   String[] columnNames = new String[]{"UserID","UserName","Sex","Remark"};
   public void handle(ColumnModel columnModel) {
      for (int i = 0; i < columnNames.length; i++) {
         if(columnModel.getColumnName().equalsIgnoreCase(columnNames[i])){
            columnModel.setColumnName(columnNames[i]);
         }
      }
   }
});

buildConfig = new ProjectBuildConfig(projectConfig);
buildConfig.setTableName("Sys_UserInfo"); 

虽然列模型处理器的只有一个handle方法,但这个简单的接口却可以实现很多灵活而强大的功能。

2.2.5 dataModel节点

dataModel指的是数据模型,它是专门用来定义在模板里面需要被动态替换的一组数据。简单来说,就是一个键值对,name为键名称,节点里面包含的字符串就是它对应的值。如下面就是一个名称为“codgen”的数据模型:

<dataModel name="codgen">myCodgen</dataModel>

前面定义的数据模型可以被后面的数据模型嵌套引用,如下面的一个数据模型里面的内容就引用了上面的“codgen”:

<dataModel name="helloCodgen">hello,${codgen}!</dataModel>

那么经过解析后,就会输出内容: hello,myCodgen! 由以上的配置文件示例可以发现,目前在配置文件里通过dataModel标签只能定义一些简单级别的数据模型,如${copyright.author}这样的复杂对象级别的数据模型就需要在编程里面这样来定义:

Copyright copyright = new copyright();
copyright.setAuthor("黄天政");
dataModelMap.put("copyright",copyright); //dataModelMap就是一棵数据模型树

像以上copyright这样的数据模型,在数据模型树里属于一级节点,包括在配置文件里面声明的都属于一级数据模型节点。codgen除了提供一个TableModel比较核心的数据模型以外,还内置了其他一些数据模型,如下列表所述:

  • projectName 项目名称,默认取配置文件里project节点的name属性的值
  • projectLabel 项目标签,即项目的中文描述信息,默认取配置文件里project节点的label属性的值
  • tableName 表名称,没有默认值,必须指定,否则不能创建表模型对象TableModel
  • tableLable 表标签,默认取表注释,若未取得则设置为表名称
  • groupName 分组名称,如果表名称符合这样的规范:Sys_UserInfo,那么默认以Sys为分组名称
  • moduleName 模块名称,如果表名称符合这样的规范:Sys_UserInfo,那么默认以UserInfo为模块名称
  • copyright 版权对象,它里面的每个属性都设有默认值,具体参考相应的API文档
  • templateDirectory 默认的模板目录,默认为相对类路径下的“template/项目名称”子目录,注意,这里的项目名称不是应用程序名称,而是配置文件里project节点的名称,譬如以上的配置项目“AOWork” ,默认的类路径为“template/AOWork”,当然这个默认值可以在配置文件里重新定义,以覆盖代码里的默认值。 注:

TableModel数据模型的名称并不是“tableModel”,而是“table”; 以上的数据模型都可以在配置文件里面直接引用,如下面一个数据模型的定义内容就引用了“copyright.author”和“projectName”两个模型:

<dataModel name="outputDirectory">D:/QtoneProject/TengenCode/${copyright.author}/${projectName}</dataModel>

2.2.6 template节点

template只有name和type两个属性,分别表示模板名称和模板的类型。模板类型也只有两个枚举值,一个是file类型;另一个是text类型。template并不是project必需的节点,一般在模板内容比较多,并且类型为text的时候才使用,当然也可以指定type为file类型,而这时节点里面的内容就是模板所在的路径,而不是模板文本内容。下面的代码片段就是一个关于版权信息的模板定义:

<template name="tModel" type="text">  
<![CDATA[  
    ${defaultCopyright}    
    ${defaultJavaModelWithNoCopyright}  
]]>  
</template> 

其中defaultCopyright和defaultJavaModelWithNoCopyright是codgen预先定义的两个数据模型。

2.2.7 output节点

output指明一个构建输出的目标,其属性及说明如下表所示:

  • name 输出名称,必须指定并要求唯一,构建输出完成后,可以根据这个名称来取得对应的输出结果,也就是一个OutputModel?对象
  • type 输出类型,也是必须指定的,目前只有file和text两种类型。file表示输出到一个文件里,这时output节点开始和结束之间的内容应该是一个文件路径;当输出类型为text时,output节点之间的内容将会被忽略,即保持空即可
  • templateName 模板名称,指向一个已定义的name属性为该名称的template节点。
  • templateFile 模板文件,该属性相当于一个类型为file,属性值为文件路径的template节点,即省去了template节点的定义。路径可以是绝对的或相对的,如果是相对路径指的是相对于classes下的目录。
  • templateContent 模板内容,该属性相当于一个类型为file,属性值为模板内容的template节点,即省去了template节点的定义。 以上属性中,其中templateName、templateFile和templateContent这三个属性都是非必需的,也就是说在这三个属性都不指定的情况下,output的类型必须为file类型,此时的模板类型也就默认为file类型,模板路径为:默认模板路径(${templateDirectory})+ 输出的名称+".ftl"。但是如果指定了其中一个或者多个属性的话,那么codgen就会按照“templateContent > templateFile > templateName”这样的适用规则来选取。

2.3 编写调用代码

2.3.1 Web环境下使用示例(以jsp为例)

//取得指定项目名称的配置信息 
ProjectConfig projectConfig = ProjectConfigHelper.getProjectConfig(getServletContext(),projectName);
//实例化一个项目构建配置对象 
ProjectBuildConfig buildConfig = new ProjectBuildConfig(projectConfig);
//设置数据库表名称(Sys_UserInfo)
buildConfig.setTableName(tableName); 
//设置表标签(如:用户信息)
buildConfig.setTableLabel(tableLabel); 
//设置分组名称(通常为表 名称的前缀,如:Sys)
buildConfig.setGroupName(groupName); 
//设置模块名称(如:UserInfo)
buildConfig.setModuleName(moduleName); 
//设置版本的创建人
buildConfig.getCopyright().setAuthor(author); 
//由一个构建配置对象实例化一个代码生成器对象
Builder builder = new CodeBuilder(buildConfig);
//执行构建操作,返回构建结果
Map<String,OutputModel> omMap = builder.build();

在web环境下,codgen是这样去寻找配置文件来加载的:如果web.xml配置了名称为“codgen.config”的上下文初始化参数(相对于WEB-INF/classes目录下的文件路径,多个文件名以英文逗号分隔);如果未配置该参数,则默认读取类路径根目录下的"codgen-config.xml",所以在这种情况下请确保src目录下存在codgen-config.xml文件。以下代码片段说明了如何在web.xml里指定加载配置:

 <context-param>
                <param-name>codgen.config</param-name>
                <param-value>codgen-config.xml,../config/codgen-Another.xml</param-value>
        </context-param>

其中“../config”为WEB-INF下的一个子目录,而“Another”是其中一个构建项目的名称,在“codgen-Another.xml”这个配置文件里只有一个project节点,名称就是“Another”。如果一个构建项目的配置信息比较多,建议按照项目名称来分开文件配置。

2.3.2 非web环境下使用示例

        public static void main(String[] args) {
                //取得默认的项目配置信息,在非web环境下调用传递参数null即可,但配置文件必须是src目录下的codgen-config.xml
                ProjectConfig projectConfig = ProjectConfigHelper.getDefaultProjectConfig(null);
                ProjectBuildConfig buildConfig = new ProjectBuildConfig(projectConfig);
                buildConfig.setTableName("Sys_UserInfo");
                //buildConfig.setTableLabel("用户信息"); //不设置默认为表注释,表注释为空时则使用表名称
                //buildConfig.setGroupName("System");  //不设置默认为Sys
                //buildConfig.setModuleName("UserInfo"); //不设置默认为UserInfo
                //buildConfig.getCopyright().setAuthor("黄天政"); //不设置默认为当前计算机用户名
                
                Builder builder = new CodeBuilder(buildConfig);
                Map<String,OutputModel> omMap = builder.build();
                for(Entry<String,OutputModel> entry : omMap.entrySet()){
                        System.out.println("生成内容="+entry.getValue().getOutput());
                }
        }
Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: You must give any other recipients of the Work or Derivative Works a copy of this License; and You must cause any modified files to carry prominent notices stating that You changed the files; and You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "{}" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright 2015 黄天政 Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.

简介

Codgen是一个基于数据库元数据模型,使用freemarker模板引擎来构建输出的代码生成器。 展开 收起
Apache-2.0
取消

发行版

暂无发行版

贡献者

全部

近期动态

不能加载更多了
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化