Hibernate 配置 双向 对多关联 (未完待续·······)
从生疏到熟练 是要经历多少遍的练习? 这答案只能向自己找。
以Student和Course为例,一个学生可以选多门课程,一门课程也可以被多个学生选取;
首先 我们创建持久化类Student
package bean;
import java.util.Set;
public class Student {
private long id;
private String name;//学生姓名
private Set<Course> courses;//该学生选择的课程
//省略set、get方法
}
接下来就是持久化类Course
package bean;
import java.util.Set;
public class Course {
private long id;
private String name;//课程名称
private Set<Student> students;//选择该课程的学生
//省略set、get方法
}
然后是对象关系映射文件Student.hbm.xml:
<hibernate-mapping>
<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>
<set name="courses" table="students_courses" cascade="save-update">
<key column="student_id"></key>
<many-to-many class="bean.Course" column="course_id"></many-to-many>
</set>
</class>
</hibernate-mapping>
多对多关联关系的实现需要一个连接表,<set>的属性指出的就是连接表的名称,<key>指出连接表参照students表id的外键的字段名;<many-to-many>中的class指定与Student多对多关联的类,column指定连接表参照Course映射表(此处由Course.hbm.xml映射为courses表)id的外键的字段名,Course.hbm.xml中的<set>配置与Student.hbm.xml中<set>相反:
Course.hbm.xml:
<hibernate-mapping>
<class name="bean.Course" table="courses">
<id name="id" column="id" type="long">
<generator class="increment"></generator>
</id>
<property name="name" column="name" type="string"></property>
<set name="students" table="students_courses" cascade="save-update" inverse="true">
<key column="course_id"></key>
<many-to-many class="bean.Student" column="student_id"></many-to-many>
</set>
</class>
</hibernate-mapping>
注意:两个映射文件中设置的连接表的名称以及连接表中的两个字段名需对应相同,如连接表名都为"students_courses"两字段为"student_id"和"course_id",否则会导致不必要的麻烦;连接表的主键为联合主键(student_id,course_id)。
三个表的结构及对应关系如下所示:


接着就是保存对象:
Student s1=new Student();
s1.setName("lisi");
Course c1=new Course();
c1.setName("English");
Course c2=new Course();
c2.setName("science");
s1.setCourses(new HashSet<Course>());
c1.setStudents(new HashSet<Student>());
c2.setStudents(new HashSet<Student>());
s1.getCourses().add(c1);
s1.getCourses().add(c2);
c1.getStudents().add(s1);
c2.getStudents().add(s1); session.save(c1);
session.save(s1);
(1)如果两个映射文件的inverse都设为false(默认),则会出现异常(主键重复)导致插入失败:
org.hibernate.exception.ConstraintViolationException: Could not execute JDBC batch update
Caused by: java.sql.BatchUpdateException: Duplicate entry '1-1' for key 'PRIMARY'
com.mysql.jdbc.exceptions.jdbc4.MySQLIntegrityConstraintViolationException:Duplicate entry '1-1' for key 'PRIMARY'
解释:应为两映射文件中的inverse都为true,则Student和Course都去维护关联关系,即同时向连接表中插入记录,则会导致主键重复而插入失败。
解决办法:
——将其中一方的inverse设为true,让对方维持关联关系;
——将s1.getCourses().add(c1);或 c1.getStudents().add(s1);删除,因为若某个Course中的students集合为空时,它就不会去向连接表中添加记录,也就不会与Student向连接表中插入记录时冲突而主键重复。
(2)如果都设为true,则都不会向连接表中插入记录而只是向两表中插入记录(两者都认为对方会维持关联关系)执行的SQl语句为:
Hibernate: insert into courses (name, id) values (?, ?)
Hibernate: insert into students (name, id) values (?, ?)
Hibernate: insert into courses (name, id) values (?, ?)
(3)设一方的inverse为true,正常插入数据时输出的SQL语句为:
Hibernate: insert into courses (name, id) values (?, ?)
Hibernate: insert into students (name, id) values (?, ?)
Hibernate: insert into courses (name, id) values (?, ?)
Hibernate: insert into students_courses (student_id, course_id) values (?, ?)
Hibernate: insert into students_courses (student_id, course_id) values (?, ?)
删除学生(Student)记录:
Student s=(Student)session.get(Student.class, 2L);
session.delete(s);
注意:
(1)如果不是Student维持关联关系:
——若连接表students_courses中有参照students表中该记录的记录(即在students_courses表中存在student_id为2L的记录)时,则删除失败。
——若连接表students_courses中没有参照students表中该记录的记录时,则可以成功地将该记录删除。
(2)如果是Student维持关联关系:
——先将连接表students_courses中参照students表中该记录的记录删除,然后将该学生记录从students表中删除
查询某学生选的所有课程:
Student s=(Student)session.get(Student.class, 2L);
Set<Course> set=s.getCourses(); for (Iterator iterator = set.iterator(); iterator.hasNext();) {
Course course = (Course) iterator.next();
System.out.println(course.getName());
某学生又选了一门新课(增加了连接表中的一条记录):
Student s=(Student)session.get(Student.class, 2L);
Course c=(Course)session.get(Course.class,1L );
s.getCourses().add(c);
c.getStudents().add(s);
删除某学生的一条选课记录(删除了连接表中的一条记录):
Student s=(Student)session.get(Student.class, 2L);
Course c=(Course)session.get(Course.class,1L );
s.getCourses().remove(c);
(此时只会删除连接表中的一条记录而不会去修改Students和courses表中的记录)
注意:双向多对多,还可以拆成两个多对一
<many-to-one name="stu" class="Student">
<column name="stuId"></column>
</many-to-one>
<many-to-one name="cou" class="Course">
<column name="couid"></column>
</many-to-one>
当遇到困难、 问题其实是好的, 只在懂不懂得努力去解决。
欢迎、感谢 您来指点我:http://www.cnblogs.com/smbk/
Hibernate 配置 双向 对多关联 (未完待续·······)的更多相关文章
- Hibernate 配置双向多对多关联
本文解决问题:Hibernate 中配置项目(Project) 员工(Employee) 双向多对多关联 方案一:直接配置双向多对多 方案二:配置第三个关联类(xml) 将多对多查分开来(形成 ...
- Hibernate二级缓存(未完待续)
1.Hibernate的cache介绍: Hibernate实现了良好的Cache机制,可以借助Hibernate内部的Cache迅速提高系统的数据读取性能.Hibernate中的Cache可分为两层 ...
- php-安装与配置-未完待续2
一,准备工作 在入门指引中,我们已经知道PHP的3个应用领域,不同的场景,需要安装的东西是不同的.具体如下: 服务器端脚本,在通常情况下,需要三样东西:PHP 自身.一个 web 服务器和一个 web ...
- Go web编程学习笔记——未完待续
1. 1).GOPATH设置 先设置自己的GOPATH,可以在本机中运行$PATH进行查看: userdeMacBook-Pro:~ user$ $GOPATH -bash: /Users/user/ ...
- MVC丶 (未完待续······)
希望你看了此小随 可以实现自己的MVC框架 也祝所有的程序员身体健康一切安好 ...
- [python]爬代理ip v2.0(未完待续)
爬代理ip 所有的代码都放到了我的github上面, HTTP代理常识 HTTP代理按匿名度可分为透明代理.匿名代理和高度匿名代理. 特别感谢:勤奋的小孩 在评论中指出我文章中的错误. REMOTE_ ...
- 阿里云服务器:IIS网站的架设(一、环境设置与安装IIS网站 二、网站的基本设置 三、建立新网站(未完待续))
Windows Server 2012 R2的Internet Information Services (IIS)网站的模块化设计,可以减少被攻击面并减轻管理负担,让系统管理员更容易架设安全的具备高 ...
- odoo11 model+Recordset 基础未完待续
Model 一个模型代表了一个业务对象 本质上是一个类,包含了同django flask一样的数据字段 所有定义在模型中的方法都可以被模型本身的直接调用 现在编程范式有所改变,不应该直接访问模型,而是 ...
- AutoMapper介绍(未完待续、部分没实现)
实体间转换工具.其实也可以用Json来实现同名属性.异名属性(用JsonProperty指明)的自动转换 最新版本6.11 需要使用vs2013以上.vs2012下载新版 nuget会遇到问题.只能旧 ...
随机推荐
- iOS越狱开发(一)
做越狱开发也有一些时间了,有很多东西想总结一下,希望给他人一些借鉴,也是自己对过去开发经历的一些总结.个人不推荐使用盗版,这里主要以技术介绍为主. 这个系列里面主要介绍怎样进行越狱开发,涉及到以下几个 ...
- javascript运算符——关系运算符
× 目录 [1]恒等 [2]相等 [3]大于[4]小于 前面的话 关系运算符用于测试两个值之间的关系,根据关系是否存在而返回true或false,关系表达式总是返回一个布尔值,通常在if.while或 ...
- java中同步嵌套引起的死锁事例代码
/* 目的:自己写一个由于同步嵌套引起的死锁! 思路:多个线程在执行时,某一时刻,0-Thread绑定了LockA锁,1-Thread绑定了LockB锁! 当0-Thread要去绑定LockB锁时 和 ...
- office快速制作简历
毕业的一年是由学校向社会转变的一年,面临着人生的一个重大转折--找工作.在如今信息爆炸的时代,纵使力拔山兮气盖世也难免会被遗落芳草之中而不得一展宏图.对未来的憧憬,对美好生活的向往,或多或少你需要一份 ...
- 献给所有从事IT行业拥有梦想的英语渣们
本文适合读者:从小到大英语都不好,如今选择IT行业需要重拾英语追寻梦想的人,英语大神们请绕行.. 目录 一.背景介绍 二.明确学习目标 2.1 英文文法 2.2 词汇量 三.题外话 3.1 关于本文 ...
- 如何使用Jedis操作Redis消息队列
资源链接 Jedis的jar包 Commons-io的jar包 使用方法 代码样例如下,使用前,注意打开redis的server程序. 代码样例 package RedisExample; impor ...
- HT for Web整合OpenLayers实现GIS地图应用
HT for Web作为逻辑拓扑图形组件自身没有GIS功能,但可以与各种GIS引擎即其客户端组件进行融合,各取所长实现逻辑拓扑和物理拓扑的无缝融合,本章将具体介绍HT for Web与开发免费的Ope ...
- 高性能Web系统设计方案(初稿目录),支持者进
第一部分 客户端篇 1.压缩js.css,将js的引入放在</html>之前; 2.合并一个页面下的js/css文件,压缩传输.(SquishIt) 相关博文 3.ajax技术应用.aja ...
- Spark API 之 combineByKey(一)
1 前言 combineByKey是使用Spark无法避免的一个方法,总会在有意或无意,直接或间接的调用到它.从它的字面上就可以知道,它有聚合的作用,对于这点不想做过多的解释,原因很简单, ...
- 修改cdh5集群中主机节点IP或hostName
前言 在使用cdh集群过程中,难免会因为某些不可抗拒的原因导致节点IP或hostName变动,而cm的监控界面无法完成这些事情,但是cm将集群中所有的主机的信息都存在postgresql数据库的hos ...