CowNew开源团队网站 [url]http://www.cownew.com[/url]
作者 杨中科 是CowNew开源团队发起人之一,邮箱about521  at 163 dot com
论坛 [url]http://www.cownew.com/newpeng/[/url]
转载请注明此版权信息

   最近准备把进销存项目激活,这样一方面可以让更多的人有机会参与到开源开发中来,另一方面也把SQL翻译器、SQL优化器、JDBMonitor应用到这个项目中,这样这三个基础模块就可以在实际项目应用中得到验证和增强。
    我准备用hibernate实现持久层,于是到hibernate的网站上把hibernate3下载下来,看了看有个hibernatetools,是个hibernate在eclipse下的辅助工具,也down了下来,用了用感觉不错。可以直接从数据库表生成POJO和hbm.xml(其实我不是很喜欢这种开发方式,我更喜欢我用建模工具来写POJO,然后用工具生成hbm.xml和DDL,不过好像hibernate3现在还没有这种工具,如果哪位朋友知道有这种工具,希望赐教)。
美中不足的是它生成的javabean的字段名是完全和数据库字段名一致、生成的javabean的类名是完全和数据库表名一致。出于清晰以及可移植的考虑(也是公司的开发规范养成的习惯),我设计的表名全部以"T_"开头,中间再加上子系统名,最后才是表意的表名,比如用“T_PS_BOM”表示生产管理系统中的物料清单表;字段名全部以“F”开头,比如FId,FName。
这样就导致生成了如下的javabean:
public class TPSBOM
{
   .......
   public String getFID()
   ...
   public String getFNumber()
}

    看起来很不直观。我刚刚想放弃这个工具,想了想,“拿来就用,不好用就换”可不是做开源人该有的精神呀。钻研一下。
看看了Hibernate Code Generation页签中有一个“reveng Strategy”,什么意思?“反向工程策略”??好像有门儿,点击“Browse”弹出一个类选择对话框,竟然看到了它默认显示的“DefaultReverseEngineeringStrategy”类了,我在hibernatetools的安装目录找来找去,终于在plugins\org.hibernate.eclipse_3.2.0.beta6\lib\tools下的hibernate-tools.jar中找到了这个类的影子,用反编译工具反编译一下(懒得去网上下源码了,呵呵)。一个个方法名展现在我面前:
tableToClassName
columnToPropertyName
columnToHibernateTypeName
。。。
    这不就是在把数据库相应的项映射成java相应的项吗?
    开工!
    新建一个类CowNewReverseEngineeringStrategy,继承自DefaultReverseEngineeringStrategy,override  tableToClassName、
columnToPropertyName这两个方法,在这两个方法中写入自己的转换逻辑。
然后打包成jar包,放到plugins\org.hibernate.eclipse_3.2.0.beta6\lib\tools下,然后在plugins\org.hibernate.eclipse_3.2.0.beta6\lib\tools\MANIFEST.MF中把这个新增包的内容加上,关闭eclipse,加个-clean参数启动eclipse,然后点击“Hibernate Code Generation”,把“reveng Strategy”填成“com.cownew.DevTools.hibtools.RevEng.CowNewReverseEngineeringStrategy”,“Run”!!!
晕倒,竟然报错“com.cownew.DevTools.hibtools.RevEng.CowNewReverseEngineeringStrategy
Exception while generating code
Reason   org.hibernate.console.HibernateConsoleRunTimeException:Could not create or find com.   with one argument deleate constructor”
看来是反射调用的时候出了问题,重新打开hibernate-tools.jar,仔细观察,竟然发现了一个DelegatingReverseEngineeringStrategy,它多    了一个参数为“ReverseEngineeringStrategy delegate”的构造函数,而其他调用都是转发给ReverseEngineeringStrategy了,晕倒,搞不懂它在做什么,也没时间研究了,给CowNewReverseEngineeringStrategy也曾街一个参数为“ReverseEngineeringStrategy delegate”的构造函数,重新打包,重新启动eclipse,哈哈,一切搞定,终于生成我可爱的,
public class PersonInfo

  public String getNumber()。。。
  public String getId()。。。

了。
    附全部代码:
package com.cownew.DevTools.hibtools.RevEng;
import java.beans.Introspector;
import org.hibernate.cfg.reveng.DefaultReverseEngineeringStrategy;
import org.hibernate.cfg.reveng.ReverseEngineeringSettings;
import org.hibernate.cfg.reveng.ReverseEngineeringStrategy;
import org.hibernate.cfg.reveng.ReverseEngineeringStrategyUtil;
import org.hibernate.cfg.reveng.TableIdentifier;
import org.hibernate.util.StringHelper;
public class CowNewReverseEngineeringStrategy extends
DefaultReverseEngineeringStrategy
{
 public CowNewReverseEngineeringStrategy(ReverseEngineeringStrategy delegate)
 {  
  super();
 }
 private ReverseEngineeringSettings settings = new ReverseEngineeringSettings();
 public String tableToClassName(TableIdentifier table)
 {
  String tableName = table.getName();
  if (tableName != null && tableName.toUpperCase().startsWith("T_"))
  {
   String pkgName = settings.getDefaultPackageName();
   int lastIndex = tableName.lastIndexOf('_');
   tableName = tableName.substring(lastIndex + 1, tableName.length())
     + "Info";
   String className = toUpperCamelCase(tableName);
   if (pkgName.length() > 0)
    return StringHelper.qualify(pkgName, className);
   return className;
  } else
  {
   return super.tableToClassName(table);
  }
 };
 public String columnToPropertyName(TableIdentifier table, String column)
 {
  if (column != null && column.toUpperCase().startsWith("F"))
  {
   String cownewColName = column.substring(1, column.length());
   
   String decapitalize = Introspector
     .decapitalize(toUpperCamelCase(cownewColName));
   return keywordCheck(decapitalize);
  } else
  {
   return super.columnToPropertyName(table, column);
  }
 }
 private String keywordCheck(String possibleKeyword)
 {
  if (ReverseEngineeringStrategyUtil
    .isReservedJavaKeyword(possibleKeyword))
   possibleKeyword += "_";
  return possibleKeyword;
 }
 public void setSettings(ReverseEngineeringSettings settings)
 {
  super.setSettings(settings);
  this.settings = settings;
 }
 public static void main(String[] args)
 {
  TableIdentifier table = new TableIdentifier("T_BD_Person");
  //TableIdentifier table = new TableIdentifier("T_Person");
  //TableIdentifier table = new TableIdentifier("Person");
  CowNewReverseEngineeringStrategy revEng = new CowNewReverseEngineeringStrategy(null);
  String className = revEng.tableToClassName(table);
  System.out.println(className);
  System.out.println(revEng.columnToPropertyName(table, "FId"));
  System.out.println(revEng.columnToPropertyName(table, "Id"));
 }
}