hibernate映射配置文件详解

Hibernate中,各表的映射文件….hbm.xml可以通过工具生成,例如在使用MyEclipse开发时,它提供了自动生成映射文件的工具。配置文件的基本结构如下:

Xml代码

1.                         <?xml version="1.0" encoding='UTF-8'?>   

2.                           

3.                         <!DOCTYPE hibernate-mapping PUBLIC   

4.                              "-//Hibernate/Hibernate Mapping DTD 3.0//EN"   

5.                              "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" >   

6.                         <hibernate-mapping package="包名">   

7.                            <class name="类名" table="表名">   

8.                               <id name="主键在java类中的字段名" column="对应表中字段" type="类型 ">   

9.                                   <generator class="主键生成策略"/>   

10.                           </id>   

11.                       

12.                               ……   

13.                         </class>   

14.                     </hibernate-mapping>   

1. 主键(id 
Hibernate
的主键生成策略有如下几种: 
1) assigned 
主键由外部程序负责生成,在 save() 之前指定。 
2) hilo 
通过hi/lo 算法实现的主键生成机制,需要额外的数据库表或字段提供高位值来源。 
3) seqhilo 
hilo 类似,通过hi/lo 算法实现的主键生成机制,需要数据库中的 Sequence,适用于支持 Sequence 的数据库,如Oracle 
4) increment 
主键按数值顺序递增。此方式的实现机制为在当前应用实例中维持一个变量,以保存着当前的最大值,之后每次需要生成主键的时候将此值加1作为主键。这种方式可能产生的问题是:不能在集群下使用。 
5) identity 
采用数据库提供的主键生成机制。如DB2SQL ServerMySQL 中的主键生成机制。 
6) sequence 
采用数据库提供的 sequence 机制生成主键。如 Oralce 中的Sequence 
7) native 
 Hibernate 根据使用的数据库自行判断采用 identityhilosequence 其中一种作为主键生成方式。 
8) uuid.hex 
 Hibernate 基于128  UUID 算法 生成16 进制数值(编码后以长度32 的字符串表示)作为主键。 
9) uuid.string 
uuid.hex 类似,只是生成的主键未进行编码(长度16),不能应用在 PostgreSQL 数据中。 
10) foreign 
使用另外一个相关联的对象的标识符作为主键。 
主键配置举例如下:

Xml代码

1.                         <id name="id" column="id" type="java.lang.Integer">   

2.                                      <generator class="native"/>   

3.                         </id>   

另外还可以扩展Hibernate的类来做自己的主键生成策略,具体例子见:http://www.javaeye.com/topic/93391 
2. 普通属性(property 
       
开发人员可以打开网址:http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd 
来查看hibernate3.0dtd信息,可看到property的定义如下:

Xml代码

1.                         <!ELEMENT property (meta*,(column|formula)*,type?)>   

2.                         <!ATTLIST property name CDATA #REQUIRED>   

3.                         <!ATTLIST property node CDATA #IMPLIED>   

4.                         <!ATTLIST property access CDATA #IMPLIED>   

5.                         <!ATTLIST property type CDATA #IMPLIED>   

6.                         <!ATTLIST property column CDATA #IMPLIED>   

7.                         <!ATTLIST property length CDATA #IMPLIED>   

8.                         <!ATTLIST property precision CDATA #IMPLIED>   

9.                         <!ATTLIST property scale CDATA #IMPLIED>   

10.                     <!ATTLIST property not-null (true|false) #IMPLIED>   

11.                     <!ATTLIST property unique (true|false) "false">   

12.                     <!ATTLIST property unique-key CDATA #IMPLIED>   

13.                     <!ATTLISTpropertyindexCDATA#IMPLIED>      

14.                     <!-- include the columns spanned by this property in an index -->   

15.                     <!ATTLIST property update (true|false) #IMPLIED>   

16.                     <!ATTLIST property insert (true|false) #IMPLIED>   

17.                     <!ATTLIST property optimistic-lock (true|false) "true">           

18.                     <!-- only supported for properties of a class (not component) -->   

19.                     <!ATTLIST property formula CDATA #IMPLIED>   

20.                     <!ATTLIST property lazy (true|false) "false">   

21.                     <!ATTLIST property generated (never|insert|always) "never">   

它的各属性中比较常用的有:name(对应的java类的属性名称)、column(对应的表中的字段)、type(属性的类型,eg.java.lang.String)、not-null(设置该属性是否为空,为true时表示非空,默认为false)和length(字段的长度限制) 示例如下:

Xml代码

1.                         <property name="accessname" column="accessName" type="java.lang.String" not-null="true" />   

2.                         <property name="state" column="state" type="java.lang.Byte" not-null="true" />   

3.                         <property name="description" column="description" type="java.lang.String" />  

3. 一对多关系(<many-to-one…/><set…></set> 

       
一对多关系一般是用在一个表与另一个表存在外键关联的时候,例如用户表的组织id与组织表存在外键关联,则方为组织表,方为用户表,因为一个组织可以包含多个用户,而一个用户只能隶属于一个组织。 

    
对于存在一对多关系和多对一关系的双方,需要在…hbm.xml中进行相应配置,这时在方(例如:组织)需要在映射文件中添加<set…></set>元素,因为它包含多个方的对象,一般的格式如下:

Xml代码

1.                         <set name="java映射类中对应的属性" inverse="true" lazy="true">   

2.                           <key column="表中对应字段"/>   

3.                              <one-to-many class="多方的类"/>   

4.                         </set>   

5.                           

6.                         <!-- 示例 -->  

7.                         <set name="userSet" inverse="true" lazy="true">   

8.                               <key column="orgId"/>   

9.                               <one-to-many class="User"/>   

10.                     </set>   

方(例如:用户)隶属于一个方对象,一般的格式如下:

Xml代码

1.                         <many-to-one name="java映射类中对应的属性" column="表中对应字段" class="类名" not-null="true" />   

2.                           

3.                         <!-- 示例 -->  

4.                         <many-to-one name="org" column="orgId" class="Organization" not-null="true" />   

4. 一对一关系(<one-to-one…/> 

一对一关系相对一对多关系来说比较少见,但也在某些情况下要用到,例如有一个用户的基本信息表(USER)和一个用户的密码表(PASSWD)就存在一对一的关系。下面来看一下一对一关系在Hibernate的配置。

Xml代码

1.                         <!-- 其中主表(eg. 用户的基本信息表)的配置 -->  

2.                           

3.                         <one-to-one name="主表对象中子表对象的属性名" class="子表对象的类名" cascade="save-update"/>   

4.                           

5.                         <one-to-one name="password" class="com.amigo.dao.pojo.Passwd" cascade="save-update"/>   

6.                           

7.                         <!-- 子表(eg. 用户的密码表)的配置 -->  

8.                           

9.                         <one-to-one name="子表对象中主表对象的属性名" class="主表对象的类名" constrained="true" />   

10.                       

11.                     <one-to-one name="user" class="com.amigo.dao.pojo.User " constrained="true" />   

5. 多对多关系(<many-to-many…/> 

在数据库设计时,一般将多对多关系转换为两个一对多(或多对一)关系,例如在基于角色的权限系统中,用户和角色存在的关系就是典型的多对多关系,即一个用户可以具有多个角色,而一个角色又可以为多个用户所有,一般在设计时,都会加一个用户与角色的关联表,该表与用户表以及角色表都存在外键关联。 

在本小节中讲述的是没有分解的多对多关系在Hibernate中如何配置。设置格式如下:

Xml代码

1.                         <set name="java对象的属性名" table="表名" cascade="all" outer-join="false">     

2.                             <key column="表的对应字段"/>     

3.                             <many-to-many class="另一个表的对象类" column="另一个表的字段"/>     

4.                         </set>     

5.                           

6.                         <!-- t_user -->  

7.                         <set name="roleSet" table="t_user" cascade="all" outer-join="false">     

8.                             <key column="roleId"/>     

9.                             <many-to-many class="com.amigo.dao.pojo.Role" column="roleId"/>     

10.                     </set>     

11.                       

12.                     <!-- t_role -->  

13.                     <set name="userSet" table="t_role" cascade="all" outer-join="false">     

14.                         <key column="roleId"/>     

15.                         <many-to-many class="com.amigo.dao.pojo.User" column="roleId"/>     

16.                     </set>

6. 完整实例 

在本小节中举一些.hbm.xml映射文件的例子,让开发人员对其有一个感性的认识。接下来讲述一个用户表(tbl_user)、用户与角色关联表(tbl_user_role)、角色表(tbl_role)以及组织表(tbl_organization)的例子。 
1tbl_userXml代码         <?xml version="1.0" encoding='UTF-8'?> <!DOCTYPE hibernate-mapping PUBLIC       "-//Hibernate/Hibernate Mapping DTD 3.0//EN"     "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" >    

     <hibernate-mapping package="com.amigo.dao.pojo">   

8.                             <class name="User" table="tbl_user">   

9.                               <id name="loginname" column="loginName" type="java.lang.String">   

10.                               <generator class="assigned"/>   </id>   

12.                          <property name="name" column="name" type="java.lang.String" not-null="true" />   

13.                          <property name="password" column="password" type="java.lang.String" not-null="true" />   

14.                          <property name="mobile" column="mobile" type="java.lang.String" />   

15.                         

<property  name= "telephone"  column= "telephone"  type= "java.lang.String"  />   

16.                          <property name="email" column="email" type="java.lang.String" />   

17.                          <property name="createtime" column="createTime" type="java.util.Date" not-null="true" />   

18.                          <property name="lastlogintime" column="lastLoginTime" type="java.util.Date" />   

19.                         <property name="logintimes" column="loginTimes" type="java.lang.Long" not-null="true" />   

20.                          <property name="state" column="state" type="java.lang.Byte" not-null="true" />   

21.                         <property name="description" column="description" type="java.lang.String" />   

22.                         <many-to-one name="organization" column="orgId" class="Organization" not-null="true" />   

23.                         <set name="userRoleSet" inverse="true" cascade="all-delete-orphan" lazy="true">   

24.                             <key column="loginName"/>   

25.                             <one-to-many class="UserRole"/> </set>   

27.                     </hibernate-mapping>


(2)tbl_organization

Xml代码

1.                         <?xml version="1.0" encoding='UTF-8'?>   

2.                           

3.                         <!DOCTYPE hibernate-mapping PUBLIC   

4.                                      "-//Hibernate/Hibernate Mapping DTD 3.0//EN"   

5.                                          "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd" >   

6.                           

7.                         <hibernate-mapping package="com.amigo.dao.pojo">   

8.                            <class name="Organization" table="tbl_organization">   

9.                                 <id name="orgid" column="orgId" type="java.lang.Long">   

10.                                <generator class="native"/>   

11.                             </id>   

12.                       

13.                             <property name="parentorgid" column="parentOrgId" type="java.lang.Long" not-null="true" />   


.                             <property name="orgname" column="orgName" type="java.lang.String" not-null="true" />   

15.                             <property name="orgfullname" column="orgFullName" type="java.lang.String" />   

16.                             <property name="orglevel" column="orgLevel" type="java.lang.Integer" not-null="true" />   

17.                             <property name="state" column="state" type="java.lang.Byte" not-null="true" />   

18.                             <property name="description" column="description" type="java.lang.String" />   

19.                             <property name="creator" column="creator" type="java.lang.String" />   

20.                             <property name="createtime" column="createTime" type="java.util.Date" />   

21.                             <set name="userSet" inverse="true" lazy="true">   

22.                                 <key column="orgId"/>   

23.                                 <one-to-many class="User"/>   

24.                             </set>   

25.                         </class>   

26.                     </hibernate-mapping>   

  1. 1、到底在哪用cascade="..."?   
  2.    cascade属性并不是多对多关系一定要用的,有了它只是让我们在插入或删除对像时更方便一些,只要在cascade的源头上插入或是删除,所有 cascade的关系就会被自己动的插入或是删除。便是为了能正确的cascade,unsaved-value是个很重要的属性。Hibernate通 过这个属性来判断一个对象应该save还是update,如果这个对象的id是unsaved-value的话,那说明这个对象不是 persistence    object要save(insert);如果id是非unsaved-value的话,那说明这个对象是persistence    object(数据库中已存在),只要update就行了。saveOrUpdate方法用的也是这个机制。     
  3.       
  4.    2、到底在哪用inverse="ture"?     
  5.     “set的inverse属性决定是否把对set的改动反映到数据库中去。inverse=false————反映;inverse=true————不反映”inverse属性默认为false  

1
  • inverse属性默认是false的,就是说关系的两端都来维护关系。这个意思就是说,如有一个Student,    Teacher和TeacherStudent表,Student和Teacher是多对多对多关系,这个关系由TeacherStudent这个表来表 现。那么什么时候插入或删除TeacherStudent表中的记录来维护关系呢?在用hibernate时,我们不会显示的对 TeacherStudent表做操作。对TeacherStudent的操作是hibernate帮我们做的。hibernate就是看hbm文件中指 定的是""维护关系,那个在插入或删除""时,就会处发对关系表的操作。前提是""这个对象已经知道这个关系了,就是说关系另一头的对象已经set 或是add到""这个对象里来了。前面说过inverse默认是false,就是关系的两端都维护关系,对其中任一个操作都会处发对表系表的操作。当在 关系的一头,如Student中的bag或set中用了inverse="true"时,那就代表关系是由另一关维护的(Teacher)。就是说当这插 入Student时,不会操作TeacherStudent表,即使Student已经知道了关系。只有当Teacher插入或删除时才会处发对关系表的 操作。所以,当关系的两头都用inverse="true"是不对的,就会导致任何操作都不处发对关系表的操作。当两端都是inverse= "false"或是default值是,在代码对关系显示的维护也是不对的,会导致在关系表中插入两次关系。     
  •       
  •    在一对多关系中inverse就更有意义了。在多对多中,在哪端inverse="true"效果差不多(在效率上)。但是在一对多中,如果要一方维护关 系,就会使在插入或是删除""方时去update""方的每一个与这个""的对象有关系的对象。而如果让""方面维护关系时就不会有update 操作,因为关系就是在多方的对象中的,直指插入或是删除多方对象就行了。当然这时也要遍历""方的每一个对象显示的操作修关系的变化体现到DB中。不管 怎样说,还是让""方维护关系更直观一些。   
  •   
  •      (1)对one-to-many而言,改变set,会让hibernate执行一系列的update语句, 不会delete/insert数据   
  •      (2)对many-to-many而言,改变set,只修改关系表的数据,不会影响many-to-many的另一方。   
  •      (3)虽然one-to-many和many-to-many的数据库操作不一样,但目的都是一个:维护数据的一致性。     
  •       
  •    3、cascade和inverse有什么区别?     
  •    可以这样理解,cascade定义的是关系两端对象到对象的级联关系;而inverse定义的是关系和对象的级联关系。   
  •    inverse只对set+one-to-many(或many-to-many)有效,对many-to-one, one-to-one无效。cascade对关系标记都有效。   
  •   
  •    inverse对集合对象整体起作用,cascade对集合对象中的一个一个元素起作用,如果集合为空,那么cascade不会引发关联操作。   
  •    比如将集合对象置为null, school.setStudentSet(null)   
  •     inverse导致hibernate执行:udpate STUDENT set SCHOOL_ID=null where SCHOOL_ID=?   
  •     cascade则不会执行对STUDENT表的关联更新, 因为集合中没有元素。   
  •    再比新增一个school, session.save(school)   
  •     inverse导致hibernate执行:   
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值