hibernate3.2(六)多对一关联映射

多个用户对应一个组,要在用户中体现出多对一,所以用户中要private Group group

public class User {
	private int id;//给实体一个唯一性的标识
	private String name;
	private Group group;
                getter and setter...
	
}

 

<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC 
	"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
	"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
    <!-- 对类进行映射,name = "完整路径" , 默认存的表和实体类的名字一致,
          可以用 table = "xxx"更改生成的数据库表-->
	<class name="com.wyx.hibernate.User" table="t_user">
		<!-- 标识 数据表默认字段和实体属性名一致,可用column="xxx"来重新定义字段名-->
		<id name="id" column="u_id">
		    <!-- 主键的生成策略 uuid全局的唯一标识,32位字符串,一般一万年不会重复-->
		    <!-- native 自动根据方言识别 -->
			<generator class="native"/>
		</id>
		<!-- 除了标识以外,都用property来映射 ,生成的字段可以重命名,unique="true"标识唯一-->
		<property name="name" column="u_name"  not-null="true" length="30"/>
		<many-to-one name="group" column="u_group" />
	</class>
</hibernate-mapping>

 省略group实体和hbm.xml,注意hbm文件中class标签中要制定table别名,因为默认生成的表group是数据库的关键字,会出错的。

 

执行测试方法:

public void testReference(){
		Session session = HibernateUtils.getSession();
		try {
			session.beginTransaction();
			
			Group group = new Group();
			group.setName("一起探索");
			
			User user1 =new User();
			user1.setName("张三");
			user1.setGroup(group);
			
			User user2 =new User();
			user2.setName("李四");
			session.save(group);
			//不能成功保存
			session.save(user1);
			System.out.println("----------mark1------------");
			session.save(user2);
			System.out.println("----------mark2------------");
			//session.save(group);
			//System.out.println("----------mark3------------");
			session.getTransaction().commit();
		} catch (HibernateException e) {
			e.printStackTrace();
			session.getTransaction().rollback();
		}finally{
			HibernateUtils.closeSession(session);
		}
	}

 

 这里会报错:

org.hibernate.TransientObjectException.

原因是:

group为transient状态 ,id我们设定为hibernate帮我们自增,id是在session.save方法后,也就是group状态变为persistent才有id,hibernate才给你自动赋值。这里group的id还没有值就被user引用了。去掉注释部分可成功运行。

结论:persistent状态对象是不能引用transient对象,这个错误不注意会经常遇到。

 

那么不存group,有没有办法能不能成功存入user呢?

通过级联操作可以。

级联是对象的连锁操作

级联适合删除、保存、修改。

 

在user.hbm.xml中设置manytoone标签的cascade属性设置为all或save-update,默认是none。

这样在保存person对象的时候,hibernate自动为我们保存group对象,这样person依赖的group数据也就存在了,就可以成功保存内容到数据库。

 

cascade属性设置为除了none以外任何有意义的值, 它将把特定的操作传递到关联对象中。这个值就代表着Hibernate基本操作的名称, persist, merge, delete, save-update, evict, replicate, lock, refresh, 以及特别的值delete-orphanall,并且可以用逗号分隔符 来组合这些操作,例如,cascade="persist,merge,evict"cascade="all,delete-orphan"

 

console打印:

Hibernate: insert into t_group (g_name) values (?)
Hibernate: insert into t_user (u_name, u_group) values (?, ?)
----------mark1------------
Hibernate: insert into t_user (u_name, u_group) values (?, ?)
----------mark2------------

 

 

下面来做个例子,从用户中加载组的信息:

 

	public void testReferenceLoad(){
		Session session = HibernateUtils.getSession();
		try {
			session.beginTransaction();
			User user = (User)session.load(User.class, 5);
			System.out.println("user.name = " + user.getName()+" user.group.name = " + user.getGroup().getName());
			session.getTransaction().commit();
		} catch (HibernateException e) {
			e.printStackTrace();
			session.getTransaction().rollback();
		}finally{
			HibernateUtils.closeSession(session);
		}
	}

 Console:

Hibernate: select group0_.g_id as g1_1_0_, group0_.g_name as g2_1_0_ from t_group group0_ where group0_.g_id=?
user.name = 张三 user.group.name = 一起探索

这里不管casecade的值设为任意的都不影响级联查询的进行,因为级联只影响增、删、改的操作。

级联查询真正起作用的是User.hbm.xml中的many-to-one标签。

 

<many-to-one>会在“多”的一方加入外键,指向“一”的一方,这个外键是由该标签下的column属性定义的,即不加column属性默认与“一”的一方的属性名字相同。

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值