hibernate3.2多对一关联映射
2011年03月01日
多个用户对应一个组,要在用户中体现出多对一,所以用户中要private Group group public class User { private int id;//给实体一个唯一性的标识 private String name; private Group group; getter and setter... }
省略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、delete、none(默认是case="none")。这样即使不先执行session.save(group)也可以成功提交内容到数据库了。
cascade属性设置为除了none以外任何有意义的值, 它将把特定的操作传递到关联对象中。这个值就代表着Hibernate基本操作的名称, persist, merge, delete, save-update, evict, replicate, lock, refresh, 以及特别的值delete-orphan和all,并且可以用逗号分隔符 来组合这些操作,例如,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标签。
会在“多”的一方加入外键,指向“一”的一方,这个外键是由该标签下的column属性定义的,即不加column属性默认与“一”的一方的属性名字相同。
2011年03月01日
多个用户对应一个组,要在用户中体现出多对一,所以用户中要private Group group public class User { private int id;//给实体一个唯一性的标识 private String name; private Group group; getter and setter... }
省略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、delete、none(默认是case="none")。这样即使不先执行session.save(group)也可以成功提交内容到数据库了。
cascade属性设置为除了none以外任何有意义的值, 它将把特定的操作传递到关联对象中。这个值就代表着Hibernate基本操作的名称, persist, merge, delete, save-update, evict, replicate, lock, refresh, 以及特别的值delete-orphan和all,并且可以用逗号分隔符 来组合这些操作,例如,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标签。
会在“多”的一方加入外键,指向“一”的一方,这个外键是由该标签下的column属性定义的,即不加column属性默认与“一”的一方的属性名字相同。