现在的位置: 首页 > 综合 > 正文

Hibernate映射Map

2018年02月04日 ⁄ 综合 ⁄ 共 4051字 ⁄ 字号 评论关闭

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

抱歉!评论已关闭.