一、项目结构如下图

二、保存学生和课程及其学生选课关系代码如下(测试类中不能再有双向关联,否则会报错,因为,都维护了中间表外键,会有中间表外键冲突,如果非要写双向关联,就需要配置中设置某一方维护主键,课程和学生映射配置文件中相应配置inverse为true和false)

package com.hjp.manytomany;

import java.util.HashSet;
import java.util.Set; /**
* Created by JiaPeng on 2016/1/2.
*/
public class Course {
private int id;
private String name; public Set<Student> getStudentSet() {
return studentSet;
} public void setStudentSet(Set<Student> studentSet) {
this.studentSet = studentSet;
} private Set<Student> studentSet = new HashSet<Student>(); @Override
public String toString() {
return "Course{" +
"id=" + id +
", name='" + name + '\'' +
'}';
} public int getId() {
return id;
} public void setId(int id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
}
}

Course

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="com.hjp.manytomany.Course" table="course">
<id name="id">
<generator class="native"></generator>
</id>
<property name="name" length="20"></property>
<set name="studentSet" table="s_c">
<key column="c_id"></key>
<many-to-many class="com.hjp.manytomany.Student" column="s_id"></many-to-many>
</set>
</class>
</hibernate-mapping>

course.hbm.xml

package com.hjp.manytomany;

import java.util.HashSet;
import java.util.Set; /**
* Created by JiaPeng on 2016/1/2.
*/
public class Student {
private int id;
private String name;
private Set<Course> courseSet = new HashSet<Course>(); @Override
public String toString() {
return "Student{" +
"id=" + id +
", name='" + name + '\'' +
", courseSet=" + courseSet +
'}';
} public Set<Course> getCourseSet() {
return courseSet;
} public void setCourseSet(Set<Course> courseSet) {
this.courseSet = courseSet;
} public int getId() {
return id;
} public void setId(int id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} }

Student

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="com.hjp.manytomany.Student" table="student">
<id name="id">
<generator class="native"></generator>
</id>
<property name="name" length="20"></property>
<set name="courseSet" table="s_c">
<key column="s_id"></key>
<many-to-many class="com.hjp.manytomany.Course" column="c_id"></many-to-many>
</set>
</class>
</hibernate-mapping>

student.hbm.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">
<hibernate-configuration>
<session-factory>
<property name="hibernate.dialect">org.hibernate.dialect.MySQLDialect</property>
<property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="hibernate.connection.url">jdbc:mysql://localhost:3306/h1</property>
<property name="hibernate.connection.username">root</property>
<property name="hibernate.connection.password">hjp123</property>
<!--设置c3p0-->
<property name="hibernate.connection.provider_class">org.hibernate.connection.C3P0ConnectionProvider</property>
<!--c3p0连接池配置信息-->
<property name="c3p0.min_size">5</property>
<property name="c3p0.max_size">20</property>
<property name="c3p0.timeout">120</property>
<property name="c3p0.idle_test_period">3000</property>
<!--显示SQL并格式化-->
<property name="hibernate.show_sql">true</property>
<property name="hibernate.format_sql">true</property>
<!--ddl生成策略,后面可以改为update-->
<property name="hibernate.hbm2ddl.auto">create</property>
<!--导入映射文件-->
<mapping resource="com/hjp/manytomany/course.hbm.xml"></mapping>
<mapping resource="com/hjp/manytomany/student.hbm.xml"></mapping>
</session-factory>
</hibernate-configuration>

hibernate.cfg.xml

 public void func3() {
Session session = HibernateUtils.getSession();
session.beginTransaction();
//创建两个学生
Student student1 = new Student();
student1.setName("张三"); Student student2 = new Student();
student2.setName("李四");
//创建三门课程
Course course1 = new Course();
course1.setName("Java"); Course course2 = new Course();
course2.setName(".Net"); Course course3 = new Course();
course3.setName("php"); //学生关联课程
//张三java\php
student1.getCourseSet().add(course1);
student1.getCourseSet().add(course3);
//李四php\.Net
student2.getCourseSet().add(course2);
student2.getCourseSet().add(course3); //课程关联学生
// course1.getStudentSet().add(student1);
// course2.getStudentSet().add(student2);
// course3.getStudentSet().add(student1);
// course3.getStudentSet().add(student2); session.save(student1);
session.save(student2);
session.save(course1);
session.save(course2);
session.save(course3);
session.getTransaction().commit();
session.close();
}

测试类中测试方法

三、解除id为1的学生和id为1的课程之间关系代码

 public void func4(){
Session session = HibernateUtils.getSession();
session.beginTransaction(); //得到id为1的学生
Student student= (Student) session.get(Student.class,1);
//得到id为1的课程
Course course= (Course) session.get(Course.class,1);
//解除学生和课程的关系
student.getCourseSet().remove(course);
course.getStudentSet().remove(student); session.getTransaction().commit();
session.close();
}

解除关系代码

结果为中间表中s_id和c_id为1的记录被删除

上述代码中也可以只用course.getStudentSet().remove(student);代码删除,因为此时维护主键的一方是课程(course.hbm.xml中<set name="studentSet" table="s_c" inverse="false">和student.hbm.xml中<set name="courseSet" table="s_c" inverse="true">)

四、恢复id为1的学生和id为1的课程关系,并解除id为1的学生和id为3的课程间关系,单向绑定,利用中间表外键由课程维护

  public void func5(){
Session session = HibernateUtils.getSession();
session.beginTransaction(); Student student= (Student) session.get(Student.class,1);
Course course1= (Course) session.get(Course.class,1);
Course course2= (Course) session.get(Course.class,3);
course2.getStudentSet().remove(student);
course1.getStudentSet().add(student); session.getTransaction().commit();
session.close();
}

解除关系及添加

五、删除id为1的学生,出现结果是id为1的学生被删了,而且中间表s_id为1的记录也删除了(上述代码中任然还是课程在维护中间表外键,执行下面代码要把双方inverse配置删除)

public void func6(){
Session session=HibernateUtils.getSession();
session.beginTransaction(); Student student= (Student) session.get(Student.class,1);
session.delete(student); session.getTransaction().commit();
session.close();
}

删除学生

六、在set节点设置cascade为delete或all-delete-orphan完成级联删除

比如id为1的学生有id为1和id为3课程,删除id为1的学生,不但中间表会删除相关记录,也会将id为1和id为3的课程删除

Hibernate之多对多的更多相关文章

  1. hibernate中多对多关联

    hibernate中多对多关联 “计应134(实验班) 凌豪” 在关系数据库中有一种常见的关系即多对多关系,例如课程和学生的关系,一个学生可以选择多门课程,同时一门课程也可以被多个学生选择, 因此课程 ...

  2. 【SSH系列】Hibernate映射 -- 多对多关联映射

         映射原理 在数据库学习阶段,我们知道,如果实体和实体之间的关系是多对多,那么我们就抽出来第三张表,第一张表和第二张表的主键作为第三表的联合主键,结合我们的hibernate,多对多关联,无论 ...

  3. Hibernate的多对多实例

    在完成了一对多的实例的基础上,继续做多对多实例.例子是老师和学生,一个老师教多个学生,一个学生也有多个老师. 文档结构如图:

  4. Hibernate 单项多对一的关联映射

    在日常开发中会出现很对多对一的情况,本文介绍hibernate中多对一的关联映射. 1.设计表结构 2.创建student对象 3.创建Grade对象 4.写hbm.xml文件 5.生成数据库表 生成 ...

  5. Hibernate的多对多映射关系

    example: 老师(teacher)和学生(Student)就是一个多对多的关系吧?老师可以有多个学生,学生也可以由多个老师,那在Hibernate中多对多是怎样实现的呢?? 在Hibernate ...

  6. Hibernate中多对多的annotation的写法(中间表可以有多个字段)

    2011-07-04 6:52 一般情况下,多对多的关联关系是需要中间表的: 情况一:如果中间表仅仅是做关联用的,它里面仅有2个外键做联合主键,则使用ManyToMany(不用写中间表的Model,只 ...

  7. hibernate关联关系 (多对多)

    hibernate的多对多 hibernate可以直接映射多对多关联关系(看作两个一对多  多对多关系注意事项 一定要定义一个主控方 多对多删除 主控方直接删除 被控方先通过主控方解除多对多关系,再删 ...

  8. hibernate之多对多关系

    hibernate的多对多hibernate可以直接映射多对多关联关系(看作两个一对多) 下面我们拿三张表来做实例 t_book_hb t_book_category_hb(桥接表) t_catego ...

  9. 如何决解项目中hibernate中多对多关系中对象转换json死循环

    先写一下原因吧!我是写的SSH项目,在项目中我遇到的问题是把分页对象(也就是pageBean对象)转化为json数据,下面为代码: public class PageBean <T>{// ...

  10. Hibernate双向多对多关联

    一.配置双向多对多关联 以Project类(项目)和Emp类(员工)为例: 1.创建Project类,并需要定义集合类型的Emp属性 public class Project { //编号 priva ...

随机推荐

  1. CLR执行模式之程序集代码的执行

    所知IL是与CPU无关的机器语言,其能访问和操作对象类型,并提供指令来创建和初始化对象,调用对象上的虚方法以及直接操作数组对象等,故可视为一种面向对象的机器语言.每种语言的存在都有其存在的价值和原因, ...

  2. [Elixir009]像GenServer一样用behaviour来规范接口

    1.Behaviour介绍 Erlang/Elixir的Behaviour类似于其它语言中的接口(interfaces),本质就是在指定behaviours的模块中强制要求导出一些指定的函数,否则编译 ...

  3. 基于Microsoft Azure、ASP.NET Core和Docker的博客系统

    欢迎阅读daxnet的新博客:一个基于Microsoft Azure.ASP.NET Core和Docker的博客系统   2008年11月,我在博客园开通了个人帐号,并在博客园发表了自己的第一篇博客 ...

  4. Linux 网络编程八(epoll应用--大并发处理)

    //头文件 pub.h #ifndef _vsucess #define _vsucess #ifdef __cplusplus extern "C" { #endif //服务器 ...

  5. Sql语句里的递归查询

    Sql语句里的递归查询 SqlServer2005和Oracle 两个版本 以前使用Oracle,觉得它的递归查询很好用,就研究了一下SqlServer,发现它也支持在Sql里递归查询举例说明:Sql ...

  6. python爬虫中文网页cmd打印出错问题解决

    问题描述 用python写爬虫,很多时候我们会先在cmd下先进行尝试. 运行爬虫之后,肯定的,我们想看看爬取的结果. 于是,我们print... 运气好的话,一切顺利.但这样的次数不多,更多地,我们会 ...

  7. 20135220谈愈敏Blog8_进程的切换和系统的一般执行过程

    进程的切换和系统的一般执行过程 谈愈敏 原创作品转载请注明出处 <Linux内核分析>MOOC课程 http://mooc.study.163.com/course/USTC-100002 ...

  8. C#出题库项目的总结(2)

    前记:好吧好吧,我好好的自我检讨,这个总结拖了这么久才来写,而且一周多没有看技术相关的东西,实在罪过,不过因为想做的事情太多,所以时间的分配确实是一个很严肃的问题,不是时间不够用,是我自己没有做好时间 ...

  9. 回顾一年的IT学习历程与大学生活

    今天是2015年8月27日,距离成为大三狗还有一个多星期,在这个不算繁忙的暑假的下午来总结一下这一年来,在IT方面的学习. 一.入门(2014.3) 我大一的专业是信息工程,信息工程听上去就是信息(I ...

  10. 喝咖啡写脚本,顺便再加一点点CSS语法糖 1.选择环境

    经过对前端开发的初步了解,大体上发现了以下几点,前端开发需要使用脚本语言,主要是JavaScript,需要Html,需要CSS,这些东西相信很多人已经很熟了.但是仅仅只是学习一点简单的JS,配合Htm ...