浅析Hibernate一对多数据关联的问题(二)

开发 后端
这里介绍Hibernate一对多数据关联。指的是单向一对多数据关联一个用户有多个地址,在用户类TUser中包含地址类TAddress集合。

Hibernate一对多数据关联。指的是单向一对多数据关联一个用户有多个地址,在用户类TUser中包含地址类TAddress集合。

Hibernate如果上手了,那么所谓的一对多,多对一,多对多,一对一这些关系,应该很快能理解.下面主要介绍Hibernate一对多的问题。

1.由TUser对象将自身的id赋给addr.user_id,这样导致addr属性值变动,在事物提交的时候,会进行update。


1)当save该用户的时候,

  1. insert into t_address  (user_id, address, zipcode, tel) value (null, "HongKong", "233123", "1123")  

2)当tx.commit()时:

  1. update t_address user_id="1"address="HongKong"zipcode="233123",tel="1123" where id=2

这样,在save user时,就会出现约束违例。

调整方法:
可以在定义数据表字段时候,不加NOT NULL约束。或者在开始为user_id随意赋一个非空值(因为还要update,不正确也没关系),或者将user_id字段从TAddress.hbm.xml中删除(本例就是这样实现)。但是这些都是权宜之计,用两条SQL语句完成一次数据库操作,性能低下。而双向一对多解决了这个问题。
下面来实现双向关联:修改配置文件 TUser.hbm.xml

  1. xml version="1.0"?> 
  2. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> 
  3. <hibernate-mapping> 
  4.     <class name="cn.blogjava.start.TUser" table="T_User" catalog="sample" 
  5.      dynamic-update="true" dynamic-insert="true" 
  6.     > 
  7.         <id name="id" type="integer"> 
  8.             <column name="id" /> 
  9.             <generator class="native" /> 
  10.         id> 
  11.         <property name="name" type="string" column="name" /> 
  12.         <property name="age" type="java.lang.Integer" column="age" /> 
  13.  
  14.         <set   
  15.             name="address"   
  16.             table="t_address"   
  17.             inverse="true" 
  18.             cascade="all"   
  19.             order-by="zipcode asc" 
  20.             > 
  21.             <key column="user_id"> 
  22.             key> 
  23.             <one-to-many class="cn.blogjava.start.TAddress" /> 
  24.         set> 
  25.     class> 
  26. hibernate-mapping> 

设定inverse="true",表明将TUser类作为被动类,将数据关联的维护工作交给关联对象TAddress来管理。
在one-to-many模型中,将many一方设为主控方有助于性能的改善。(让总理记住每个人困难,但是每个人记住总理方便)

TAddress.hbm.xml

  1. xml version="1.0"?> 
  2. "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> 
  3. <hibernate-mapping> 
  4.     <class name="cn.blogjava.start.TAddress" table="T_Address" catalog="sample"> 
  5.         <id name="id" type="integer"> 
  6.             <column name="id" /> 
  7.             <generator class="native" /> 
  8.         id> 
  9.         <property name="address" type="string" column="address" /> 
  10.         <property name="zipcode" type="string" column="zipcode" /> 
  11.         <property name="tel" type="string" column="tel" /> 
  12.         <property name="type" type="string" column="type" /> 
  13.         <property name="idx" type="java.lang.Integer" column="idx" /> 
  14.         <many-to-one 
  15.                   name="user"   
  16.                   class="cn.blogjava.start.TUser" 
  17.                   cascade="none" 
  18.                   outer-join="auto" 
  19.                   update="true"                    
  20.                   insert="true" 
  21.                   access="property" 
  22.                   column="user_id" 
  23.                   not-null="true" 
  24.         /> 
  25.     class> 
  26. hibernate-mapping> 

2.对TAddress.java做如下改造:去掉user_id字段,增加user字段,和getter,setter方法。

  1. package cn.blogjava.start;  
  2.  
  3. import java.io.Serializable;  
  4.  
  5. public class TAddress implements Serializable {  
  6.       
  7.     private Integer id;  
  8.     private String address;  
  9.     private String zipcode;  
  10.     private String tel;  
  11.     private String type;  
  12.     private Integer idx;  
  13.     private TUser user;  
  14.       
  15.     public TUser getUser() {  
  16.         return user;  
  17.     }  
  18.     public void setUser(TUser user) {  
  19.         this.user = user;  
  20.     }  
  21.     public Integer getId() {  
  22.         return id;  
  23.     }  
  24.     public void setId(Integer id) {  
  25.         this.id = id;  
  26.     }  
  27.     public String getAddress() {  
  28.         return address;  
  29.     }  
  30.     public void setAddress(String address) {  
  31.         this.address = address;  
  32.     }  
  33.     public Integer getIdx() {  
  34.         return idx;  
  35.     }  
  36.     public void setIdx(Integer idx) {  
  37.         this.idx = idx;  
  38.     }  
  39.     public String getTel() {  
  40.         return tel;  
  41.     }  
  42.     public void setTel(String tel) {  
  43.         this.tel = tel;  
  44.     }  
  45.     public String getType() {  
  46.         return type;  
  47.     }  
  48.     public void setType(String type) {  
  49.         this.type = type;  
  50.     }  
  51.     public String getZipcode() {  
  52.         return zipcode;  
  53.     }  
  54.     public void setZipcode(String zipcode) {  
  55.         this.zipcode = zipcode;  
  56.     }  
  57.  
  58. }  

3.测试代码
既然TUser不维护关联关系,需要TAddress需要自己来维护TUser,所以需要addr.setUser(user);

  1. package cn.blogjava.start;  
  2.  
  3. import java.util.HashSet;  
  4. import java.util.Iterator;  
  5. import java.util.List;  
  6.  
  7. import junit.framework.Assert;  
  8. import junit.framework.TestCase;  
  9.  
  10. import org.hibernate.HibernateException;  
  11. import org.hibernate.Session;  
  12. import org.hibernate.SessionFactory;  
  13. import org.hibernate.Transaction;  
  14. import org.hibernate.cfg.Configuration;  
  15.  
  16.  
  17. public class HibernateTest extends TestCase {  
  18.       
  19.     Session session = null;  
  20.  
  21.     protected void setUp() {  
  22.         try {  
  23.             Configuration config = new Configuration().configure();  
  24.             SessionFactory sessionFactory = config.buildSessionFactory();  
  25.             session = sessionFactory.openSession();  
  26.               
  27.         } catch (HibernateException e) {  
  28.             e.printStackTrace();  
  29.         }          
  30.     }  
  31.  
  32.     protected void tearDown() {  
  33.         try {  
  34.             session.close();          
  35.         } catch (HibernateException e) {  
  36.             e.printStackTrace();  
  37.         }          
  38.     }      
  39.       
  40.     /** *//**  
  41.      * 对象持久化测试(Insert方法)  
  42.      */          
  43.     public void testInsert() {  
  44.         Transaction tran = null;  
  45.         try {  
  46.           
  47.             TUser user = new TUser();  
  48.             user.setName("byf");  
  49.             user.setAge(new Integer(26));  
  50.               
  51.             TAddress addr = new TAddress();  
  52.             addr.setTel("1123");  
  53.             addr.setZipcode("233123");  
  54.             addr.setAddress("HongKong");  
  55.             addr.setUser(user);  
  56.               
  57.             TAddress addr2 = new TAddress();  
  58.             addr2.setTel("139");  
  59.             addr2.setZipcode("116001");  
  60.             addr2.setAddress("dalian");         
  61.             addr2.setUser(user);  
  62.  
  63.             TAddress addr3 = new TAddress();  
  64.             addr3.setTel("136");  
  65.             addr3.setZipcode("100080");  
  66.             addr3.setAddress("beijing");  
  67.             addr3.setUser(user);  
  68.               
  69.             //设置关联  
  70.             HashSet set = new HashSet();  
  71.             set.add(addr);  
  72.             set.add(addr2);  
  73.             set.add(addr3);  
  74.             user.setAddress(set);  
  75.                                      
  76.             tran = session.beginTransaction();                                  
  77.             //插入user信息  
  78.             session.save(user);  
  79.             session.flush();  
  80.             tran.commit();  
  81.             Assert.assertEquals(user.getId().intValue()>0 ,true);  
  82.         } catch (HibernateException e) {  
  83.             e.printStackTrace();  
  84.             Assert.fail(e.getMessage());  
  85.             if(tran != null) {  
  86.                 try {  
  87.                     tran.rollback();  
  88.                 } catch (Exception e1) {  
  89.                     e1.printStackTrace();  
  90.                 }  
  91.             }  
  92.         }  
  93.     }  
  94.       
  95.     /** *//**  
  96.      * 对象读取测试(Select方法)  
  97.      */              
  98.     public void testSelect(){  
  99.         String hql = " from TUser where name='byf'";  
  100.         try {  
  101.             List userList = session.createQuery(hql).list();  
  102.             TUser user = (TUser)userList.get(0);  
  103.             System.out.println("user name is " + user.getName());  
  104.               
  105.             for (Iterator iter = user.getAddress().iterator(); iter.hasNext();) {  
  106.                 TAddress addr = (TAddress) iter.next();  
  107.                 System.out.println("user address is " + addr.getAddress());                  
  108.             }  
  109.             Assert.assertEquals(user.getName(), "byf");  
  110.         } catch (Exception e) {  
  111.             e.printStackTrace();  
  112.             Assert.fail(e.getMessage());  
  113.         }  
  114.     }  

以上介绍Hibernate一对多数据关联。

【编辑推荐】

  1. 生成Hibernate Mapping文件的分析
  2. 对Hibernate中get()与load()不同点分析
  3. Struts-Spring-Hibernate案例
  4. 简述Hibernate配置连接池
  5. 浅析Hibernate一对多数据关联的问题(一)
责任编辑:仲衡 来源: baidu
相关推荐

2009-09-23 10:37:50

Hibernate一对

2012-03-21 11:43:41

JavaHibernate

2012-02-08 13:34:08

HibernateJava

2009-09-22 09:55:58

Hibernate实例

2009-06-04 10:34:19

Hibernate一对一对多关系配置

2009-06-04 16:14:22

Hibernate一对Hibernate一对Hibernate多对

2009-06-03 16:27:27

Hibernate一对一关系

2010-04-15 09:09:02

Hibernate

2009-08-17 10:34:51

NHibernate一

2009-06-03 16:18:16

Hibernate关系代码实例

2009-09-22 17:32:38

Hibernate A

2009-06-17 14:55:26

Hibernate数据

2009-06-24 07:58:52

Hibernate多数

2010-07-07 08:33:09

SQL Server学

2009-09-28 17:23:51

Hibernate E

2009-12-23 09:31:11

宽带路由上网故障

2009-09-21 13:31:10

Hibernate 3

2012-02-23 09:24:52

大数据云计算

2018-04-11 15:33:59

大数据块链去中心化

2009-09-21 18:13:11

Hibernate S
点赞
收藏

51CTO技术栈公众号