【原创翻译】使用Hibernate持久化java 5 的枚举类型【AppFuse 2.0 文档-11】

使用Hibernate持久化Java5枚举类型的实践指南
本文提供了一个实例,展示了如何利用Hibernate框架将Java5的枚举类型映射到数据库中,包括书写JPA注释、定义枚举类及属性映射等关键步骤。
摘要:
本文使用两个例子展示了如何使用Hibernate来持久化java 5 的枚举类型(Enum )
Java 5 Enums Persistence with Hibernate
标题
【原创翻译】使用Hibernate持久化java 5 的枚举类型【AppFuse 2.0 文档-11】
 
作者: Added by eboz, last edited by eboz on Jun 05, 2007 17:43
翻译者: 陈海青(josonchen)
日期:2007.11.22
 
摘要:
本文使用两个例子展示了如何使用Hibernate来持久化java 5 的枚举类型(Enum )
关键字:
Appfuse 2.0 , java 5,enum, annotations , Hibernate,
 
英文版网址 http://appfuse.org/display/APF/Java+5+Enums+Persistence+with+Hibernate
 
中文版
-------
关于本教程
如果你需要使用 JPA 的注释( annotation )将 Java 5 的枚举类型( Enum )映射到 Hibernate ,那么本文就是你需要的 .
目录
书写JPA 注释(annotations)
首先,使用一个来自 AppFuse 持久化教程的经典 POJO 的例子来展示如何使用注释 (annotation).
package org.appfuse.tutorial.model;
 
@Entity
@Table(name="t_person")
public class Person extends BaseObject {
 
   // Enumerations ---------------------------
     public enum Sex{
 
      MALE(1),
      FEMALE(2);
 
      private int value;
 
      Sex(int value) {
          this.value = value;
      }
 
      // the identifierMethod
      public int toInt() {
        return value;
      }
 
       // the valueOfMethod
       public  static Sex fromInt(int value) {    
           switch(value) {
               case 1: return MALE;
               case 2: return FEMALE;
               default:
                       return UNKNOW;
           }
      }
     
      public String toString() {
        switch(this) {
          case MALE:
              return "Male";
          case FEMALE:
              return "Female";
        }
      }
    }
 
    // Attributes ---------------------------
 
    @Id
    @Column(name= "person_id")
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long id;
 
 
    @Column(name="person_firstname", length = 50, nullable = false)
    private String firstName;
 
 
    @Column(name="person_lastname", length = 50, nullable = false)
    private String lastName;
 
   
    @Column(name= "person_sex", columnDefinition="integer", nullable = true)
    @Type(
        type = "org.appfuse.tutorial.commons.hibernate.GenericEnumUserType",
        parameters = {
                @Parameter(
                    name = "enumClass",                      
                    value = "org.appfuse.tutorial.model.Person$Sex"),
                @Parameter(
                    name = "identifierMethod",
                    value = "toInt"),
                @Parameter(
                    name = "valueOfMethod",
                    value = "fromInt")
                }
    )
    private Sex sex;
 
 
    /*
     * Getters and Setters ...
     */
}
把Java 5 的枚举类型(Enum)映射到Hibernate
现在,使用以下的灵活方案 (Flexible solution) – 一个可运行版本( working version 来映射 Sex 属性。这里要感谢 Martin Kersten, 他的文章 Java 5 EnumUserType 提供了帮助 .
package org.appfuse.tutorial.commons.hibernate;
 
import java.io.Serializable;
import java.lang.reflect.Method;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.Properties;
 
import org.hibernate.HibernateException;
import org.hibernate.type.NullableType;
import org.hibernate.type.TypeFactory;
import org.hibernate.usertype.ParameterizedType;
import org.hibernate.usertype.UserType;
 
 
public class GenericEnumUserType implements UserType, ParameterizedType {
    private static final String DEFAULT_IDENTIFIER_METHOD_NAME = "name";
    private static final String DEFAULT_VALUE_OF_METHOD_NAME = "valueOf";
 
    private Class<? extends Enum> enumClass;
    private Class<?> identifierType;
    private Method identifierMethod;
    private Method valueOfMethod;
    private NullableType type;
    private int[] sqlTypes;
 
    public void setParameterValues(Properties parameters) {
        String enumClassName = parameters.getProperty("enumClass");
        try {
            enumClass = Class.forName(enumClassName).asSubclass(Enum.class);
        } catch (ClassNotFoundException cfne) {
            throw new HibernateException("Enum class not found", cfne);
        }
 
        String identifierMethodName = parameters.getProperty("identifierMethod", DEFAULT_IDENTIFIER_METHOD_NAME);
 
        try {
            identifierMethod = enumClass.getMethod(identifierMethodName, new Class[0]);
            identifierType = identifierMethod.getReturnType();
        } catch (Exception e) {
            throw new HibernateException("Failed to obtain identifier method", e);
        }
 
        type = (NullableType) TypeFactory.basic(identifierType.getName());
 
        if (type == null)
            throw new HibernateException("Unsupported identifier type " + identifierType.getName());
 
        sqlTypes = new int[] { type.sqlType() };
 
        String valueOfMethodName = parameters.getProperty("valueOfMethod", DEFAULT_VALUE_OF_METHOD_NAME);
 
        try {
            valueOfMethod = enumClass.getMethod(valueOfMethodName, new Class[] { identifierType });
        } catch (Exception e) {
            throw new HibernateException("Failed to obtain valueOf method", e);
        }
    }
 
    public Class returnedClass() {
        return enumClass;
    }
 
    public Object nullSafeGet(ResultSet rs, String[] names, Object owner) throws HibernateException, SQLException { 
        Object identifier = type.get(rs, names[0]);
        if (identifier == null) {
            return null;
        }
        
        try {
            return valueOfMethod.invoke(enumClass, new Object[] { identifier });
        } catch (Exception e) {
            throw new HibernateException("Exception  + valueOfMethod.getName() + "' of " +while invoking valueOf method '"
                    "enumeration class '" + enumClass + "'", e);
        }
    }
 
    public void nullSafeSet(PreparedStatement st, Object value, int index) throws HibernateException, SQLException {
        try {
            if (value == null) {
                st.setNull(index, type.sqlType());
            } else {
                Object identifier = identifierMethod.invoke(value, new Object[0]);
                type.set(st, identifier, index);
            }
        } catch (Exception e) {
            throw new HibernateException("Exception  + identifierMethod.getName() + "' of " +while invoking identifierMethod '"
                    "enumeration class '" + enumClass + "'", e);
        }
    }
 
    public int[] sqlTypes() {
        return sqlTypes;
    }
 
    public Object assemble(Serializable cached, Object owner) throws HibernateException {
        return cached;
    }
 
    public Object deepCopy(Object value) throws HibernateException {
        return value;
    }
 
    public Serializable disassemble(Object value) throws HibernateException {
        return (Serializable) value;
    }
 
    public boolean equals(Object x, Object y) throws HibernateException {
        return x == y;
    }
 
    public int hashCode(Object x) throws HibernateException {
        return x.hashCode();
    }
 
    public boolean isMutable() {
        return false;
    }
 
    public Object replace(Object original, Object target, Object owner) throws HibernateException {
        return original;
    }
}
Delphi 12.3 作为一款面向 Windows 平台的集成开发环境,由 Embarcadero Technologies 负责其持续演进。该环境以 Object Pascal 语言为核心,并依托 Visual Component Library(VCL)框架,广泛应用于各类桌面软件、数据库系统及企业级解决方案的开发。在此生态中,Excel4Delphi 作为一个重要的社区开源项目,致力于搭建 Delphi 与 Microsoft Excel 之间的高效桥梁,使开发者能够在自研程序中直接调用 Excel 的文档处理、工作表管理、单元格操作及宏执行等功能。 该项目以库文件与组件包的形式提供,开发者将其集成至 Delphi 工程后,即可通过封装良好的接口实现对 Excel 的编程控制。具体功能涵盖创建与编辑工作簿、格式化单元格、批量导入导出数据,乃至执行内置公式与宏指令等高级操作。这一机制显著降低了在财务分析、报表自动生成、数据整理等场景中实现 Excel 功能集成的技术门槛,使开发者无需深入掌握 COM 编程或 Excel 底层 API 即可完成复杂任务。 使用 Excel4Delphi 需具备基础的 Delphi 编程知识,并对 Excel 对象模型有一定理解。实践中需注意不同 Excel 版本间的兼容性,并严格遵循项目文档进行环境配置与依赖部署。此外,操作过程中应遵循文件访问的最佳实践,例如确保目标文件未被独占锁定,并实施完整的异常处理机制,以防数据损毁或程序意外中断。 该项目的持续维护依赖于 Delphi 开发者社区的集体贡献,通过定期更新以适配新版开发环境与 Office 套件,并修复已发现的问题。对于需要深度融合 Excel 功能的 Delphi 应用而言,Excel4Delphi 提供了经过充分测试的可靠代码基础,使开发团队能更专注于业务逻辑与用户体验的优化,从而提升整体开发效率与软件质量。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值