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