Hibernate的多对一和一对多操作实例

开发 后端
Hibernate的一对多和多对一操作是非常方便的,如果在减少数据库复杂度的原则来说,把一些逻辑处理放在持久层,还是一个比较常见的方法。

Hibernate的一对多和多对一操作真的很方便,如果系统采用Hibernate作为持久层,完全可以把对应的一对多和多对一逻辑关系放在Hibernate里面控制,减少数据库的负担,而且也更清晰。

1、多对一和一对多概念

其实这个概念上来说很简单,比如一个客户可以有多个订单,多个订单属于同一个客户。就是最基本的一对多,和多对一。数据库使用中,感觉多对一和一对多算是比较常见的逻辑关系了。

我曾经做过一些数据库,比如某些政府部门的,其表单很设计的很简单粗糙,甚至连主键都没有,完全靠在事务层补全这些关系。其实通过Hibernate持久层来实现逻辑关系也是很不错的方法。下面的例子,就是数据库逻辑上基本没有定义,主要放在持久层里面。这个也主要是我对数据库操作属于半通水的原因。

2、数据库层

这里面有两个表单,一个CUSTOMER,客户表单,一个是ORDERS,订单表单。生成客户表单,这个是在SQLServer里面做的,其实其他都一样,因为逻辑关系在Hibernate上面,id是主键非空,其他可以为空:

  1. CREATETABLE[dbo].[CUSTOMER](  
  2. [id][numeric](18,0)NOTNULL,  
  3. [name][varchar](50)NULL,  
  4. [age][int]NULL,  
  5. CONSTRAINT[PK_CUSTOMER]PRIMARYKEY) 

订单表单

id为主键非空,CUSTOMER_id是对应客户主键,也非空,这里不做外键设置。

  1. CREATETABLE[dbo].[ORDERS](  
  2. [id][numeric](18,0)NULLPRIMARYKEY,  
  3. [CUSTOMER_id][numeric](18,0)NOTNULL,  
  4. [ORDER_NUMBER][varchar](50)NULL,  
  5. [PRICE][numeric](18,3)NULL  

3、Hibernate设定

HIbernate里面,一对多的对象体现,是客户有一个集合set,set里面放着对应订单,而多对一体现,是订单里面有一个CUSTOMER对象,表明该订单所属的客户。其中,CUSTOMER类为:

  1. publicclassCustomerimplementsjava.io.Serializable{  
  2. privateLongid;  
  3. privateStringname;  
  4. privateIntegerage;  
  5. privateSetrderses=newHashSet();  
  6.  
  7. } 

后面的getXXX和setXXX方法就省去了,同样订单类就是:

  1. publicclassOrdersimplementsjava.io.Serializable{  
  2. privateLongid;  
  3. privateCustomercustomer;  
  4. privateStringorderNumber;  
  5. privateDoubleprice;  
  6.  
  7. }  

而对应hbm文档,就是map文档如下:

  1. CUSTOMER.hbm.xml  
  2. <!DOCTYPEhibernate-mappingPUBLIC"-//Hibernate/HibernateMappingDTD3.0//EN"  
  3. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> 
  4. <!--  
  5. MappingfileautogeneratedbyMyEclipsePersistenceTools  
  6. --> 
  7. <hibernate-mapping> 
  8. <classnameclassname="onetomany.Customer"table="CUSTOMER"schema="dbo"catalog="DBTEST"> 
  9. <idnameidname="id"type="java.lang.Long"> 
  10. <columnnamecolumnname="id"precision="18"scale="0"/> 
  11. <generatorclassgeneratorclass="increment"/> 
  12. </id> 
  13. <propertynamepropertyname="name"type="java.lang.String"> 
  14. <columnnamecolumnname="name"length="50"/> 
  15. </property> 
  16. <propertynamepropertyname="age"type="java.lang.Integer"> 
  17. <columnnamecolumnname="age"/> 
  18. </property> 
  19. <setnamesetname="orderses"inverse="true"lazy="true"cascade="all"> 
  20. <key> 
  21. <columnnamecolumnname="CUSTOMER_id"precision="18"scale="0"not-null="true"/> 
  22. </key> 
  23. <one-to-manyclassone-to-manyclass="onetomany.Orders"/> 
  24. </set> 
  25. </class> 
  26. </hibernate-mapping> 

#p#
这个里面,其他都很简答了,其中<generatorclass="increment"/>表示主键值自动增加,这个主要针对字符串对应的,主要体现多对以的是:

  1. <setnamesetname="orderses"inverse="true"lazy="true"cascade="all"> 
  2. <key> 
  3. <columnnamecolumnname="CUSTOMER_id"precision="18"scale="0"not-null="true"/> 
  4. </key> 
  5. <one-to-manyclassone-to-manyclass="onetomany.Orders"/> 
  6. </set> 

其中,set表示,对应集合;fetch和lazy主要是用来级联查询的,而cascade和inverse主要是用来级联插入和修改的,这几个主要包括对集合的控制。<one-to-manyclass="onetomany.Orders"/>表示对应类,即set里面包含的类,而key主要是用于确定set里面对应表单列。

  1. ORDERS的hbm  
  2. <?xmlversionxmlversion="1.0"encoding="utf-8"?> 
  3. <!DOCTYPEhibernate-mappingPUBLIC"-//Hibernate/HibernateMappingDTD3.0//EN"  
  4. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> 
  5. <!--  
  6. MappingfileautogeneratedbyMyEclipsePersistenceTools  
  7. --> 
  8. <hibernate-mapping> 
  9. <classcatalogclasscatalog="DBTEST"name="onetomany.Orders"schema="dbo"table="ORDERS"> 
  10. <idnameidname="id"type="java.lang.Long"> 
  11. <columnnamecolumnname="id"precision="18"scale="0"/> 
  12. <generatorclassgeneratorclass="increment"/> 
  13. </id> 
  14. <many-to-oneclassmany-to-oneclass="onetomany.Customer"fetch="select"name="customer"> 
  15. <columnnamecolumnname="CUSTOMER_id"precision="18"scale="0"/> 
  16. </many-to-one> 
  17. <propertygeneratedpropertygenerated="never"lazy="false"name="orderNumber"type="java.lang.String"> 
  18. <columnlengthcolumnlength="50"name="ORDER_NUMBER"/> 
  19. </property> 
  20. <propertygeneratedpropertygenerated="never"lazy="false"name="price"type="java.lang.Double"> 
  21. <columnnamecolumnname="PRICE"precision="18"scale="3"/> 
  22. </property> 
  23. </class> 
  24. </hibernate-mapping> 
  25. <many-to-oneclassmany-to-oneclass="onetomany.Customer"fetch="select"name="customer"> 
  26. <columnnamecolumnname="CUSTOMER_id"precision="18"scale="0"/> 
  27. </many-to-one> 

表示CUSTOMER熟悉对应的类,和其作为key的列名,上面这些都可以在MyEclipse里面自动生成。另外注意的一点是,在生成的DAO里面,涉及表单操作的save()和delete()方法,必须要事件提交,数据库才有反映。可以就该Hibernate.xml,或者用下面这样代码来实现:

  1. Sessionse=getSession();  
  2. Transactiontx=se.beginTransaction();  
  3. se.delete(persistentInstance);  
  4. //se.save(instance);  
  5. tx.commit(); 

4、验证效果

1、新增用户

如果新增一个用户,该用户里面包含有两个表单,那么,由于持久层已经实现了逻辑关系,只要用户类里面的set包含了表单,则表单可以自动增加。实现代码:

  1. CustomerDAOcd=newCustomerDAO();  
  2. Customerxd=newCustomer("王小虎",20,null);  
  3. Ordersord1=newOrders();  
  4. ord1.setCustomer(xd);  
  5. ord1.setOrderNumber("王小虎的买单1");  
  6. Ordersord2=newOrders();  
  7. ord2.setCustomer(xd);  
  8. ord2.setOrderNumber("王小虎的买单2");  
  9. Setrderses=newHashSet();  
  10. orderses.add(ord1);  
  11. orderses.add(ord2);  
  12. xd.setOrderses(orderses);  
  13. cd.save(xd); 

代码里面,加入一个王小虎用户。两个订单,通过setOrderses加入,只使用cd.save这一个对持久层操作。完成后查询:

  1. 王小虎  
  2. =================================  
  3. 王小虎的买单1  
  4. 王小虎的买单2 

显示,CUSTOMER里面加入了王小虎,ORDERS里面也加入他的订单。

2、删除操作

  1. List<Customer>csList=cd.findByProperty("name","王小虎");  
  2. for(Customercs:csList){  
  3. cd.delete(cs);  

这个很简单了,通过其中findByProperty("name","王小虎");对应SQL为deletefromtableCUSTOMERwherename=''王小虎';删除了王小虎,而ORDERS里面,王小虎对应的表单也同时被删除。

5、小小总结

Hibernate的多对一和一对多处理,还是挺方便的,如果在减少数据库复杂度的原则来说,把一些逻辑处理放在持久层是一个常见的方法。

【编辑推荐】

  1. Hibernate批量更新与删除实例浅析
  2. 简述Hibernate Synchronizer学习笔记
  3. Hibernate column属性简介
  4. 概括Hibernate查询语言
  5. Hibernate cartridge学习总结
责任编辑:王晓东 来源: ITPUB个人空间
相关推荐

2009-09-22 09:55:58

Hibernate实例

2009-06-04 16:14:22

Hibernate一对Hibernate一对Hibernate多对

2009-07-21 17:31:39

iBATIS一对多映射

2023-06-12 08:09:01

FlaskSQLAlchemy

2009-06-03 16:18:16

Hibernate关系代码实例

2022-02-18 11:05:25

Jpa配置Address

2009-06-04 10:34:19

Hibernate一对一对多关系配置

2012-03-21 11:43:41

JavaHibernate

2009-06-03 16:27:27

Hibernate一对一关系

2009-06-18 14:22:06

Hibernate多对Hibernate

2009-09-23 10:37:50

Hibernate一对

2010-07-07 08:33:09

SQL Server学

2012-02-08 13:34:08

HibernateJava

2009-09-23 10:57:02

Hibernate一对

2009-06-24 15:51:47

Entity Bean一对多映射

2011-03-25 10:49:30

Join一对多

2010-09-10 13:45:18

SQLCOUNT()函数

2011-08-17 10:28:53

多对多查询SQL Server

2019-05-12 14:10:07

物联网DDOS网络攻击

2009-07-21 17:39:50

iBATIS的多对多映
点赞
收藏

51CTO技术栈公众号