首先两个类,一个是班级类,一个是学生类:
public class Grade{private int id;private String name;private Set students = new HashSet();}public class Student {private int id;private String studentName;}
数据库中表的结构:
t_grade: 两个字段:id name
t_student: 三个字段:id studentName gradeid
Grade类的映射文件:Grade.hbm.xml(此时是单向关联)
<hibernate-mapping><class name="Grade" table="t_grade" lazy="false"><id name="id"><generator class="native"/></id><property name="name"/><strong><set name="students" cascade="save-update" lazy="false"><key column=" gradeid "/><one-to-many class="Student"/></set></strong></class> </hibernate-mapping>
现在执行以下java代码:
Set students = new HashSet();Student s1 = new Student ();s1.setStudentName("s1"); Student o2 = new Student();s2.setStudentName ("s2"); students.add(s1);students.add(s2);Grade g = new Grade();g.setName("g1");g1.setStudents(students); session.save(c);
此时Hibernate发出的sql语句如下:
Hibernate: insert into t_grade (name) values (?)Hibernate: insert into t_student (studentName) values (?)Hibernate: insert into t_student (studentName) values (?)Hibernate: update t_student set gradeid=? where id=?Hibernate: update t_student set gradeid =? where id=?
此时查询数据库,显示如下:
t_grade:
id |name
1 g1
t_student:
id |studentName | gradeid
1 s1 1
2 s2 2
在保存Grade对象时,会先发出insert into t_grade(name) values (?)语句将g1同步到数据库中,因为在<set>映射中设置了cascade=”save-update”,所以会同时保存s1、s2两个对象。如果在映射文件中没有设置cascade=”save-update”的话,那么Hibernate就不会自动保存students集合对象,那在更新时将抛出异常:
Hibernate: insert into t_grade (name) values (?)Hibernate: update t_student set gradeid=? where id=?org.hibernate.TransientObjectException: Student
异常分析:
因为在<set>的inverse属性中默认是“inverse=false”,就是由Grade对象当主控方,由它负责关联关系的维护,也就是负责更新t_student表中的gradeid,但是因为没有设置casade=”save-update”,所以students集合中的对象不会在保存grade时自动保存,所以会抛出异常。
重新设置casade=”save-update”,并且设置inverse=”true”,如下:<set name="students" cascade="save-update" inverse="true" lazy="false"><key column="gradeid"/><one-to-many class="Student "/></set>
同样执行上述java代码,hibernate发出的语句如下:
Hibernate: insert into t_grade (name) values (?)Hibernate: insert into t_student (studentName) values (?)Hibernate: insert into t_student (studentName) values (?)
此时查看数据库,显示如下:
t_grade:
id |name
1 g1
t_student:
id |studentName | gradeid
1 s1 NULL
2 s2 NULL
这时t_student表的gradeid为NULL,因为设置了inverse=”true”,所以此时Student是主控方,关联关系的维护由Student来完成,在保存Grade对象时,grade不会再维护Students的gradeid属性,必须由student自己来维护,也就是说必须要设置student.setGrade(grade);
当通过Student来维护关联关系时,那么这个关联关系就转换成了双向关联。
在Student类中添加一行代码,变成如下:
public class Student { private int id; private String studentName;private Grade grade;}
此时Student.hbm.xml也要增加几行代码:
<hibernate-mapping><class name="Student" table="t_student"><id name="id"><generator class="native"/></id><property name="studentName"/> <many-to-one name="grade" column="gradeid"/></class> </hibernate-mapping>
此时如果再执行之前的java代码,hibernate发出如下语句:
Hibernate: insert into t_grade (name) values (?)Hibernate: insert into t_student (studentName,gradeid) values (?, ?)Hibernate: insert into t_student (studentName,gradeid) values (?, ?)
此时保存Student对象会为gradeid赋值,因为Student对象中拥有Grade属性,对应gradeid字段,,此时查看数据库,还是和之前的一样:
t_grade:
id |name
1 g1
t_student:
id |studentName | gradeid
1 s1 NULL
2 s2 NULL
其实,每个人都是幸福的。只是,你的幸福,常常在别人眼里。