Hibernate映射Map,比较简单的情况就是Map<key,value>的value为String、Long、Date等Hibernate支持的类,较为复杂的就是自定义的类。下面介绍映射Map时这两种情况,至于更加复杂的情况以后再说。
以Team类为例:
1.简单情况:Team中含有类型为Map<Long,String>的students属性,Long对应学生的学号,String对应学生的姓名。Team类如下:
package bean; import java.util.Map; public class Team { private long id;//Team的id private String name;//Team的名字 private Map<Long,String> students//Team中包含多个学生,Long对应学号,String对应学生姓名 //setXxx()及getXxx()方法省略 }
则对象关系映射文件为:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <!-- bean.Team对应数据库中的team表 --> <class name="bean.Team" table="team"> <!-- team表主键自增1 --> <id name="id" column="id" type="long"> <generator class="increment"/> </id> <!-- Team的名字 --> <property name="name" column="name" type="string"></property> <!-- 对应于Team类的Map<Long,String>类型的students属性 --> <map name="students" table="students"> <key column="team_id"></key><!-- students表通过team_id与team表关联 --> <index column="cardID" type="long"></index><!-- 对应于Map<key,value>中的key --> <element column="student_name" type="string"></element><!-- 对应于Map<key,value>中的value --> </map> </class> </hibernate-mapping>
对<map>的说明:Hibernate会将Team对象的Map类型的students存放在另一张表中(students表)表中的记录即<key,value>的信息,此处<map>中的<key>子元素指出students表的team_id字段为students表参照team表id字段的外键(Foreign key)。两表的结构如下:
students表中的主键为(team_id,cardID),为组合主键;因为students表中team_id相同的记录可以有多个(同一个Team对象中students属性中的多个<cardID, student_name >对), cardID 相同的记录也可有多个(不同的Team对象)。
通过hibernate API保存删除更新Team对象会同时改变team和students表的信息。
2.如果students属性的类型为Map<Long,Student>,value的类型为Student类型,,就较为复杂些了,改变如下:
修改Team类的students属性及其set、get方法:
private Map<Long,Student> students;
修改Team.hbm.xml文件:
<map name="students" table="students" cascade="all"> <key column="team_id"></key> <index column="cardID" type="long"></index> <one-to-many class="bean.Student"/><!--对应Map<key,value>中的value--> </map>
编写持久化类Student:
package bean; public class Student { private long id; private long cardID; private String name; private int age; private Team team; //getXxx()、setXxx()方法省略 }
创建映射文件:Student.hbm.xml并将其加入到hibernate.cfg.xml主配置文件中<mapping resource="Student.hbm.xml"/>
<class name="bean.Student" table="students"> <id name="id" column="id" type="long"> <generator class="increment"></generator> </id> <property name="name" column="name" type="string"></property> <property name="age" column="age" type="int"></property> <property name="cardID" column="cardID" type="long"></property> <many-to-one name="team" class="bean.Team" column="team_id" cascade="none"></many-to-one> </class>
说明:
(1).Team.hbm.xml文件中的<map ...... table="studentTT">指定的表可以不存在,因为最终的表名是由Student.hbm.xml指定的;
(2).以上Team与Student通过<map>建立了一对多的双向关系,由Student的team属性可得到其对应的Team对象。
(3).Student已定义了id属性,因此students表中的主键为id而不再是先前的(team_id,cardID)组合主键。students表中的team_id字段参照team表中的主键id,两表结构如下:
(4)即使Student类中没有定义team属性(即两者之间是单向一对多),则Hibernate依然会在students表中加入一个标识Team对象的字段team_id(在<map>的<key>中指定),以便按照team_id查找students表中的与之对应的零至多个记录并将它们填充到team_id指定的Team对象的students属性中。
保存Team对象:
tx=session.beginTransaction(); Team team=new Team(); team.setName("team1"); team.setStudents(new HashMap<Long,Student>()); team.getStudents().put(100106L, new Student(100103L,23,"A"));//观察下面的SQL语句和表中的记录验证到底谁决定该记录在students表中的cardID字段值 team.getStudents().put(100102L, new Student(100102L,24,"A2")); session.save(team); tx.commit();
控制台输出SQL语句(使用Hibernate自动建表):
Hibernate: select max(id) from team Hibernate: select max(id) from students Hibernate: insert into team (name, id) values (?, ?) Hibernate: insert into students (name, age, cardID, team_id, id) values (?, ?, ?, ?, ?)//此句插入的Student的cardID为100103L Hibernate: insert into students (name, age, cardID, team_id, id) values (?, ?, ?, ?, ?) Hibernate: update students set team_id=?, cardID=? where id=?//此句将先前的Student的cardID更新为100106L,由此可知最终的cardID是由put时的key决定的 Hibernate: update students set team_id=?, cardID=? where id=?
删除Team对象:
Team t=(Team)session.get(Team.class, 2L);//此处2L不能写为2,否则出错,因为Team类的id是long类型的 session.delete(t);
如果Team.hbm.xml中的<map>的属性cascade的值为"save-update",则设置该级联级别时,不能级联删除对应的students表中的记录,此时这些记录的某些字段值会被修改为NULL,
cardID为Map的key,team_id为students表参照Team表id字段的外键,可将cascade设为"all"或其他值,进行级联删除。
转载请注明出处:http://blog.csdn.net/jialinqiang/article/details/8683117