Hibernate对自定义类型UserType的用法

系统 1631 0

Hibernate允许我们自定义映射属性的类型,比如一个学生有联系地址,而联系地址又分为家庭地址和工作地址,我们可以把两个地址信息抽象成一个新的Address类,作为Student的成员变量

数据库结构:

 

create   table  typestu (id  varchar ( 32 primary   key ,name  varchar ( 32 ),homeaddr  varchar ( 32 ),workaddr  varchar ( 32 ));

Hibernate.cfg.xml

 

<? xml version='1.0' encoding='UTF-8' ?>
<! DOCTYPE hibernate-configuration PUBLIC
          "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
          "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd"
>

<!--  Generated by MyEclipse Hibernate Tools.                    -->
< hibernate-configuration >

< session-factory >
    
< property  name ="connection.username" > root </ property >
    
< property  name ="connection.url" >
        jdbc:mysql://localhost:3306/schoolproject?characterEncoding=gb2312
&amp; useUnicode=true
    
</ property >
    
< property  name ="dialect" >
        org.hibernate.dialect.MySQLDialect
    
</ property >
    
< property  name ="myeclipse.connection.profile" > mysql </ property >
    
< property  name ="connection.password" > 1234 </ property >
    
< property  name ="connection.driver_class" >
        com.mysql.jdbc.Driver
    
</ property >
    
< property  name ="hibernate.dialect" >
        org.hibernate.dialect.MySQLDialect
    
</ property >
    
< property  name ="hibernate.show_sql" > true </ property >
    
< property  name ="current_session_context_class" > thread </ property >
    
< mapping  resource ="Search/UserType/Student.hbm.xml"   />

</ session-factory >

</ hibernate-configuration >

 Pojo

 

package  Search.UserType;


public   class  Student  ... {
    
private  String id;  // 标识id
     private  String name;  // 学生姓名
     private  AddressType address; // 地址
     public  String getId()  ... {
        
return  id;
    }

    
public   void  setId(String id)  ... {
        
this .id  =  id;
    }

    
public  String getName()  ... {
        
return  name;
    }

    
public   void  setName(String name)  ... {
        
this .name  =  name;
    }

    
public  AddressType getAddress()  ... {
        
return  address;
    }

    
public   void  setAddress(AddressType address)  ... {
        
this .address  =  address;
    }


 
 
 
}

 

自定义类型

 


package  Search.UserType;

import  java.io.Serializable;
import  java.sql.PreparedStatement;
import  java.sql.ResultSet;
import  java.sql.SQLException;
import  java.sql.Types;

import  org.apache.commons.lang.builder.EqualsBuilder;
import  org.apache.commons.lang.builder.HashCodeBuilder;
import  org.hibernate.HibernateException;
import  org.hibernate.usertype.UserType;

public   class  AddressType  implements  UserType, Serializable  ... {
    
private  String homeAddr;
    
private  String workAddr;

    
/**/ /*  有几个字段就有几个值,这里容易出错,要多注意  */
    
private   static   final   int [] SQL_TYPES  =   ... { Types.VARCHAR, Types.VARCHAR } ;

    
/**/ /*  这个方法告诉Hibernate在成生DDL时对列采用什么样的SQL语法  */
    
public   int [] sqlTypes()  ... {
        
return  SQL_TYPES;
    }


    
/**/ /*
     * Hibernate返回什么样的映射类型,与 <property name="address" type="model.AddressType">
     * 指定的类一致。事实上也可以把AddressType拆分为两个类,一个类是只携带信息的JavaBean,它里面
     * 没有逻辑操作也没有实现UserType(比如AddressBean);而另一个类实现了UserType,它所面对的就不是现在这个
     * AddressType类的homeAddr和homeAddr属性,它面对的是AddressBean。在本例中为了简洁方便,只用了一个类。
     
*/

    
public  Class returnedClass()  ... {
        
return  AddressType. class ;
    }


    
/**/ /*
     * 表明这个类的实例在创建以后就不可以改变属性。Hibernate能为不可改变的类作一些性能优化。
     
*/

    
public   boolean  isMutable()  ... {
        
return   false ;
    }


    
/**/ /*
     * 由于AddressType是不可变的,所以深拷贝可以直接返回对象引用。拷贝的对象由应用程序使用, 而原版对象由Hibernate维护以做脏数据检查
     
*/

    
public  Object deepCopy(Object value)  ... {
        
return  value;  //  Address is immutable
    }


    
/**/ /*  两个对象是否相等,使用了apache的common工具包来进行属性比对  */
    
public   boolean  equals(Object x, Object y)  ... {
        
if  (x  ==  y)
            
return   true ;
        
if  (x  ==   null   ||  y  ==   null )
            
return   false ;
        AddressType add1 
=  (AddressType) x;
        AddressType add2 
=  (AddressType) y;
        
return   new  EqualsBuilder()  // 使用EqualsBuilder类来方便地进行比对
                .append(add1.getHomeAddr(), add2.getHomeAddr()).append(
                        add2.getWorkAddr(), add2.getWorkAddr()).isEquals();
    }


    
/**/ /*  得到hash码  */

    
public   int  hashCode(Object x)  throws  HibernateException  ... {
        AddressType address 
=  (AddressType) x;
        
return   new  HashCodeBuilder() // 使用HashCodeBuilder类来方便地进行比对
                .append(address.getHomeAddr()).append(address.getWorkAddr())
                .toHashCode();
    }


    
/**/ /*  读取数据并组装成一个AddressType对象。names[]中的参数顺序依照映射文件中定义的顺序  */
    
public  Object nullSafeGet(ResultSet rs, String[] names, Object owner)
            
throws  HibernateException, SQLException  ... {
        
if  (rs.wasNull())
            
return   null ;
        String homeAddr 
=  rs.getString(names[ 0 ]);
        String schoolAddr 
=  rs.getString(names[ 1 ]);
        AddressType address 
=  

Hibernate对自定义类型UserType的用法


更多文章、技术交流、商务合作、联系博主

微信扫码或搜索:z360901061

微信扫一扫加我为好友

QQ号联系: 360901061

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描下面二维码支持博主2元、5元、10元、20元等您想捐的金额吧,狠狠点击下面给点支持吧,站长非常感激您!手机微信长按不能支付解决办法:请将微信支付二维码保存到相册,切换到微信,然后点击微信右上角扫一扫功能,选择支付二维码完成支付。

【本文对您有帮助就好】

您的支持是博主写作最大的动力,如果您喜欢我的文章,感觉我的文章对您有帮助,请用微信扫描上面二维码支持博主2元、5元、10元、自定义金额等您想捐的金额吧,站长会非常 感谢您的哦!!!

发表我的评论
最新评论 总共0条评论