最近做项目时,使用Hibernate Tools 3.2.4生成entity和hbm.xml,但默认情况下,DB中的comments没法生成到javadoc和xml中,改了templates倒是有注释了,但却是乱码,心里一直耿耿于怀...(这不符合咱一直强调的编码规范不是?最主要的是人懒,有时用entity不想再找文档)。在网上找了半天,大多说是freemarker编码设置问题,但不管怎么设置,都没一点效果,决定自己动手。下了源码,查到原因,人家压根就没处理中文问题。记录一下处理过程。
ftl是freemarker模板,可以在jar包外使用,java和properties重新打包替换hibernate-tools.jar,如果是eclipse-plugins,jar包在plugins\org.hibernate.eclipse_3.2.4.GA-R200905070146-H18\lib\tools\hibernate-tools.jar
pojo\PojoFields.ftl
 <#-- // Fields -->
<#-- // Fields -->


 <#foreach field in pojo.getAllPropertiesIterator()><#if pojo.getMetaAttribAsBool(field, "gen-property", true)>    /** *//** *//** *//**
<#foreach field in pojo.getAllPropertiesIterator()><#if pojo.getMetaAttribAsBool(field, "gen-property", true)>    /** *//** *//** *//**
 <#if pojo.hasMetaAttribute(field, "field-description")>
<#if pojo.hasMetaAttribute(field, "field-description")>
 ${pojo.getFieldJavaDoc(field, 0)}
     ${pojo.getFieldJavaDoc(field, 0)}
 </#if>
</#if>
 <#foreach column in field.columnIterator><#if column.comment?exists && column.comment?trim?length!=0>     * ${column.comment}.
<#foreach column in field.columnIterator><#if column.comment?exists && column.comment?trim?length!=0>     * ${column.comment}.
 </#if>
</#if>
 </#foreach>
</#foreach>
 */
     */

 $
    $ {pojo.getFieldModifiers(field)} $
{pojo.getFieldModifiers(field)} $ {pojo.getJavaTypeName(field, jdk5)} $
{pojo.getJavaTypeName(field, jdk5)} $ {field.name}<#if pojo.hasFieldInitializor(field, jdk5)> = $
{field.name}<#if pojo.hasFieldInitializor(field, jdk5)> = $ {pojo.getFieldInitialization(field, jdk5)}</#if>;
{pojo.getFieldInitialization(field, jdk5)}</#if>;
 </#if>
</#if>
 </#foreach>
</#foreach>
pojo\PojoPropertyAccessors.ftl
 <#-- // Property accessors -->
<#-- // Property accessors -->
 <#foreach property in pojo.getAllPropertiesIterator()>
<#foreach property in pojo.getAllPropertiesIterator()>
 <#if pojo.getMetaAttribAsBool(property, "gen-property", true)>
<#if pojo.getMetaAttribAsBool(property, "gen-property", true)>

 /** *//**
    /** *//**  
 <#if pojo.hasFieldJavaDoc(property)>
<#if pojo.hasFieldJavaDoc(property)>    
 * ${pojo.getFieldJavaDoc(property, 4)}
     * ${pojo.getFieldJavaDoc(property, 4)}
 </#if>
</#if>
 <#foreach column in property.columnIterator><#if column.comment?exists && column.comment?trim?length!=0>     * 取得 ${column.comment}.
<#foreach column in property.columnIterator><#if column.comment?exists && column.comment?trim?length!=0>     * 取得 ${column.comment}.
 </#if>
</#if>
 </#foreach>
</#foreach>
 */
     */
 <#include "GetPropertyAnnotation.ftl"/>
    <#include "GetPropertyAnnotation.ftl"/>

 $
    $ {pojo.getPropertyGetModifiers(property)} $
{pojo.getPropertyGetModifiers(property)} $ {pojo.getJavaTypeName(property, jdk5)} $
{pojo.getJavaTypeName(property, jdk5)} $ {pojo.getGetterSignature(property)}()
{pojo.getGetterSignature(property)}()  {
{

 return this.$
        return this.$ {property.name};
{property.name};
 }
    }


 /** *//**
    /** *//**
 <#if pojo.hasFieldJavaDoc(property)>
<#if pojo.hasFieldJavaDoc(property)>    
 * ${pojo.getFieldJavaDoc(property, 4)}
     * ${pojo.getFieldJavaDoc(property, 4)}
 </#if>
</#if>
 <#foreach column in property.columnIterator><#if column.comment?exists && column.comment?trim?length!=0>     * 设置 ${column.comment}.
<#foreach column in property.columnIterator><#if column.comment?exists && column.comment?trim?length!=0>     * 设置 ${column.comment}.
 </#if>
</#if>
 </#foreach>
</#foreach>
 */
     */   

 $
    $ {pojo.getPropertySetModifiers(property)} void set$
{pojo.getPropertySetModifiers(property)} void set$ {pojo.getPropertyName(property)}($
{pojo.getPropertyName(property)}($ {pojo.getJavaTypeName(property, jdk5)} $
{pojo.getJavaTypeName(property, jdk5)} $ {property.name})
{property.name})  {
{

 this.$
        this.$ {property.name} = $
{property.name} = $ {property.name};
{property.name};
 }
    }
 </#if>
</#if>
 </#foreach>
</#foreach>

org\hibernate\tool\hbm2x\TemplateProducer.java

 public void produce(Map additionalContext, String templateName, File destination, String identifier, String fileType, String rootContext)
public void produce(Map additionalContext, String templateName, File destination, String identifier, String fileType, String rootContext)  {
{
 
        
 String tempResult = produceToString( additionalContext, templateName, rootContext );
        String tempResult = produceToString( additionalContext, templateName, rootContext );
 
        

 if(tempResult.trim().length()==0)
        if(tempResult.trim().length()==0)  {
{
 log.warn("Generated output is empty. Skipped creation for file " + destination);
            log.warn("Generated output is empty. Skipped creation for file " + destination);
 return;
            return;
 }
        }
 
        FileWriter fileWriter = null;
 Writer fileWriter = null;
        Writer fileWriter = null;

 try
        try  {
{
 
            
 th.ensureExistence( destination );
            th.ensureExistence( destination );    
 
         
 ac.addFile(destination, fileType);
            ac.addFile(destination, fileType);
 log.debug("Writing " + identifier + " to " + destination.getAbsolutePath() );
            log.debug("Writing " + identifier + " to " + destination.getAbsolutePath() );
 
            fileWriter = new FileWriter(destination);
 fileWriter = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(destination), "UTF-8"));
            fileWriter = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(destination), "UTF-8"));

 fileWriter.write(tempResult);
            fileWriter.write(tempResult);            
 }
        } 

 catch (Exception e)
        catch (Exception e)  {
{
 throw new ExporterException("Error while writing result to file", e);
            throw new ExporterException("Error while writing result to file", e);    

 } finally
        } finally  {
{

 if(fileWriter!=null)
            if(fileWriter!=null)  {
{

 try
                try  {
{
 fileWriter.flush();
                    fileWriter.flush();
 fileWriter.close();
                    fileWriter.close();
 }
                }

 catch (IOException e)
                catch (IOException e)  {
{
 log.warn("Exception while flushing/closing " + destination,e);
                    log.warn("Exception while flushing/closing " + destination,e);
 }
                }                
 }
            }
 }
        }
 
        
 }
    }
org\hibernate\tool\hbm2x\jtidy.properties
 indent=auto
indent=auto
 indent-spaces=4
indent-spaces=4
 #indent-attributes=yes
#indent-attributes=yes
    wrap=180
 markup=yes
markup=yes
 clean=yes
clean=yes
 output-xml=yes
output-xml=yes
 input-xml=yes
input-xml=yes
 show-warnings=yes
show-warnings=yes
 trim-empty-elements=yes
trim-empty-elements=yes
 input-encoding=utf-8
input-encoding=utf-8
 output-encoding=utf-8
output-encoding=utf-8
补充:
无法取得MySQL5 Table的Comments,修改org.hibernate.cfg.reveng.dialec.MySQLMetaDataDialect
重载getTables方法

 /** *//**
/** *//**
 * MetaData中无法取得table Comment,重载
     * MetaData中无法取得table Comment,重载
 */
     */
 @Override
    @Override

 public Iterator getTables(String xcatalog, String xschema, String xtable)
    public Iterator getTables(String xcatalog, String xschema, String xtable)  {
{

 try
        try  {
{
 final String catalog = caseForSearch(xcatalog);
            final String catalog = caseForSearch(xcatalog);
 final String schema = caseForSearch(xschema);
            final String schema = caseForSearch(xschema);
 final String table = caseForSearch(xtable);
            final String table = caseForSearch(xtable);

 log.debug("getTables(" + catalog + "." + schema + "." + table + ")");
            log.debug("getTables(" + catalog + "." + schema + "." + table + ")");


 ResultSet tableRs = getMetaData().getTables(catalog, schema, table, new String[]
            ResultSet tableRs = getMetaData().getTables(catalog, schema, table, new String[]  { "TABLE", "VIEW" });
{ "TABLE", "VIEW" });


 return new ResultSetIterator(tableRs, getSQLExceptionConverter())
            return new ResultSetIterator(tableRs, getSQLExceptionConverter())  {
{

 Map element = new HashMap();
                Map element = new HashMap();


 protected Object convertRow(ResultSet tableResultSet) throws SQLException
                protected Object convertRow(ResultSet tableResultSet) throws SQLException  {
{
 element.clear();
                    element.clear();
 putTablePart(element, tableResultSet);
                    putTablePart(element, tableResultSet);
 element.put("TABLE_TYPE", tableResultSet.getString("TABLE_TYPE"));
                    element.put("TABLE_TYPE", tableResultSet.getString("TABLE_TYPE"));

 String remarks = tableResultSet.getString("REMARKS");
                    String remarks = tableResultSet.getString("REMARKS");

 if (StringHelper.isEmpty(remarks))
                    if (StringHelper.isEmpty(remarks))  {
{
 String sql = "show table status " + (schema == null ? "" : " from " + schema + " ") + " like '"
                        String sql = "show table status " + (schema == null ? "" : " from " + schema + " ") + " like '"
 + element.get("TABLE_NAME") + "' ";
                                + element.get("TABLE_NAME") + "' ";
 PreparedStatement statement = getConnection().prepareStatement(sql);
                        PreparedStatement statement = getConnection().prepareStatement(sql);

 ResultSet tableRs = statement.executeQuery();
                        ResultSet tableRs = statement.executeQuery();


 if (tableRs.next())
                        if (tableRs.next())  {
{
 remarks = tableRs.getString("COMMENT");
                            remarks = tableRs.getString("COMMENT");
 }
                        }
 }
                    }
 element.put("REMARKS", remarks);
                    element.put("REMARKS", remarks);
 return element;
                    return element;
 }
                }


 protected Throwable handleSQLException(SQLException e)
                protected Throwable handleSQLException(SQLException e)  {
{
 // schemaRs and catalogRs are only used for error reporting if
                    // schemaRs and catalogRs are only used for error reporting if
 // we get an exception
                    // we get an exception
 String databaseStructure = getDatabaseStructure(catalog, schema);
                    String databaseStructure = getDatabaseStructure(catalog, schema);
 throw getSQLExceptionConverter().convert(
                    throw getSQLExceptionConverter().convert(
 e,
                            e,
 "Could not get list of tables from database. Probably a JDBC driver problem. "
                            "Could not get list of tables from database. Probably a JDBC driver problem. "
 + databaseStructure, null);
                                    + databaseStructure, null);
 }
                }
 };
            };

 } catch (SQLException e)
        } catch (SQLException e)  {
{
 // schemaRs and catalogRs are only used for error reporting if we get an exception
            // schemaRs and catalogRs are only used for error reporting if we get an exception
 String databaseStructure = getDatabaseStructure(xcatalog, xschema);
            String databaseStructure = getDatabaseStructure(xcatalog, xschema);
 throw getSQLExceptionConverter().convert(e,
            throw getSQLExceptionConverter().convert(e,
 "Could not get list of tables from database. Probably a JDBC driver problem. " + databaseStructure,
                    "Could not get list of tables from database. Probably a JDBC driver problem. " + databaseStructure,
 null);
                    null);
 }
        }
 }
    }
外键默认生成List,修改org.hibernate.cfg.JDBCBinder

 /** *//**
/** *//**
 * @param rc
     * @param rc
 * @param processed
     * @param processed
 * @param table
     * @param table
 * @param object
     * @param object
 */
     */

 private Property bindOneToMany(PersistentClass rc, ForeignKey foreignKey, Set processed, Mapping mapping)
    private Property bindOneToMany(PersistentClass rc, ForeignKey foreignKey, Set processed, Mapping mapping)  {
{

 Table collectionTable = foreignKey.getTable();
        Table collectionTable = foreignKey.getTable();
 
        Collection collection = new org.hibernate.mapping.Set(rc); // MASTER TODO: allow overriding collection type
 Collection collection = new org.hibernate.mapping.Bag(rc); // MASTER TODO: allow overriding collection type
        Collection collection = new org.hibernate.mapping.Bag(rc); // MASTER TODO: allow overriding collection type
。。。