Hibernate实体关系映射——双边的多对一、一对多关系

来源:互联网 发布:淘宝卖家在哪找淘小二 编辑:程序博客网 时间:2024/06/05 09:47
//刘梦冰发表于2015-6-18

 

双边的多对一、一对多关系

 

     双边关系是最常见的配置。在多方有一方的引用,一方也有多方的引用。双边关系能够很方便的查询数据。下面以一个班级与学生的双边关系来深入学习。

 

     班级(clazz类)与学生(student类)为一对多关系。班级中有学生属性,学生中有班级属性,二者是双边的关系。

 

具体代码如下:

 

Clazz类中有一个id主键和一个name属性,另外还有一个List<Student>类型的集合属性,代表该班级内的学生。班级与学生的关系是一对多关系,这里使用了@OneToMany配置,并用mappedBy属性配置与该List<Student>属性对应的是Student类中的clazz属性。

 

Clazz.java

 

package com.lmb.hibernate.bean;import java.util.ArrayList;import java.util.List;import javax.persistence.*;@Entity@Table(name="tb_clazz")public class Clazz {@Id@GeneratedValue(strategy = GenerationType.AUTO)private Integer id;private String name;@OneToMany(mappedBy = "clazz")private List<Student> students = new ArrayList<Student>();public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public List<Student> getStudents() {return students;}public void setStudents(List<Student> students) {this.students = students;}}


 

    双边关系中,控制权一般交给多方,因此这里的@OneToMany没有哦诶之数据库的外键列,而只配置了一个mappedBy属性,值为clazz,告诉hibernate,配置信息要到Student类的clazz属性中找。

 

  Student类中有一个id主键以及name姓名、sex性别两个普通属性,还有一个Clazz类型的属性clazz,代表所在的班级。该属性与Clazz中的List<Student>集合属性是对应的。从逻辑上来讲,某student的clazz属性为某clazz,该clazz的List<Student>中必定还有改student。配置了mappedBy后,hibernate能自动维护这种关系。

 

   Student与Clazz是一对多的关系,一对多使用@ManyToOne配置,另外用@JoinColumn配置关联的列名。这里用tb_student表的class_id列于tb_class的id主键列连接。这里同时配置了级联保存类型为PERSIST,创建student时,会自动将clazz也写进数据库。

 

student.java

 

package com.lmb.hibernate.bean;import javax.persistence.*;@Entity@Table(name="tb_student")public class Student {@Id@GeneratedValue(strategy = GenerationType.AUTO)private Integer id;private String name;private String sex;@ManyToOne(cascade = {CascadeType.PERSIST},fetch = FetchType.EAGER)@JoinColumn(name="class_id")private Clazz clazz;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getSex() {return sex;}public void setSex(String sex) {this.sex = sex;}public Clazz getClazz() {return clazz;}public void setClazz(Clazz clazz) {this.clazz = clazz;}}


注意:@JoinColumn配置了外键列,这里把控制权交给了多方,也可以交给一方。当一方控制时,一方的配置同单边的一对多关系配置完全相同。

 

    将Clazz与Studet实体类配置到hibernate配置文件中去。

 

hibernate.cfg.xml

 

<?xml version='1.0' encoding='UTF-8'?><!DOCTYPE hibernate-configuration PUBLIC          "-//Hibernate/Hibernate Configuration DTD 3.0//EN"          "http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd"><!-- Generated by MyEclipse Hibernate Tools.                   --><hibernate-configuration>    <session-factory>        <property name="dialect">org.hibernate.dialect.MySQLDialect</property>        <property name="connection.url">jdbc:mysql://localhost:3306/hibernate?characterEncoding=UTF-8</property>        <property name="connection.username">lmb</property>        <property name="connection.password">lmb</property>        <property name="connection.driver_class">com.mysql.jdbc.Driver</property>        <mapping class="com.lmb.hibernate.bean.Person" />        <mapping class="com.lmb.hibernate.bean.Email" />                <mapping class="com.lmb.hibernate.bean.Clazz" />        <mapping class="com.lmb.hibernate.bean.Student" />        </session-factory></hibernate-configuration>


 

    双边关系使用起来比较方便,查询或者使用时可以从任意一方访问到另一方。不过双边关系也不是任何时候都适用的。如果多方的数据量比较大,比如几千条几万条,设置一方的一对多映射可能是个隐患。比如,不小心执行了getStudent().size(),程序会加载这几千条几万条的子数据,造成巨大的IO开支,影响到程序的运行效率。这种情况下设置双边关系是不明智的,应该取消一方的一对多映射,不要使用getStudent().size()来统计子数据,而改用执行select count(s) from Student s查询语句的方式统计个数。

 

 

 

 

 

 

 

 

 

 

 

1 0
原创粉丝点击