一个学生可以选多门课

一门课程有多个学生上

实现步骤:

一、学生

(1)数据库创建学生数据表students,包含id,name字段

设置id字段为主键,类型:bigint,自增

设置name字段,类型:nvarchar(50)

(2)创建Student.java实体类,对应数据表

package com.zit.entities;

import java.util.HashSet;
import java.util.Set; import javax.persistence.Column;
import javax.persistence.GeneratedValue;
import javax.persistence.Id; public class Student { private long id; private String name; private Set<Course> courses = new HashSet<Course>(); public Student(){ } public long getId() {
return id;
} public void setId(long id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Set<Course> getCourses() {
return courses;
} public void setCourses(Set<Course> courses) {
this.courses = courses;
} }

  

(3)创建映射文件Student.hbm.xml,表明映射关系

1、Student.hbm.xml

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping>
<class name="com.zit.entities.Student" table="students">
<id name="id" column="id" type="long" >
<generator class="native"/>
</id>
<property name="name" column="name" type="string"/>
<!-- 添加中间表映射 -->
<set name="courses" table="students_courses" cascade="save-update" inverse="false">
<key column="student_id"></key>
<many-to-many class="com.zit.entities.Course" column="course_id"></many-to-many>
</set>
</class>
</hibernate-mapping>
()<class>的name:实体类路径, table:数据表名字

()id的<generator class="native"/>:主键生成策略,因为我在数据表已让主键id字段自增,这里设置native(别的百度一下即可)

()id的column:对应数据表字段id,数据表为bigint类型,对应Java的long类

()<set>标签:设置中间表

  table:中间表表名,

  cascade:什么情况下进行关联操作,这里设为save-update,所以save时会进行关联操作

  inverse:true/false,由false的一方维护,这里是多对多,双方维护,双方都为false

  <key column>:对应中间表的字段student_id,指定为students表的id的外键

  <many-to-many>:class:它指出与Student类进行关联的类,即Course; column:指定为courses表的主键id的外键


Course.hbm.xml中与之相反即可,具体见下面具体配置

二、课程

(1)数据库创建课程数据表courses,包含id,name字段(与学生表相同即可)

设置id字段为主键,类型:bigint,自增

设置name字段,类型:nvarchar(50)

(2)创建Course.java实体类,对应数据表

package com.zit.entities;

import java.util.HashSet;
import java.util.Set; public class Course { private long id; private String name; private Set<Student> students = new HashSet<Student>(); public Course(){ } public long getId() {
return id;
} public void setId(long id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public Set<Student> getStudents() {
return students;
} public void setStudents(Set<Student> students) {
this.students = students;
} }

  

(3)创建映射文件Course.hbm.xml,表明映射关系

<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping>
<class name="com.zit.entities.Course" table="courses">
<id name="id" column="id" type="long" >
<generator class="native"/>
</id>
<property name="name" column="name" type="string"/>
<!-- 添加中间表映射 -->
<set name="students" table="students_courses" cascade="save-update" inverse="false">
<key column="course_id"></key>
<many-to-many class="com.zit.entities.Student" column="student_id"></many-to-many>
</set>
</class>
</hibernate-mapping>

  

三、中间表

由Hibernate自动生成中间表,不需自己创建

它是根据Student.hbm.xml和Course.hbm.xml中的映射自动生成对应字段,

student_id字段,是students表的主键id的外键

course_id字段,是courses表的主键id的外键

如下图所示:

四、 测试代码:

(运行后自动生成中间表)

1、一个学生,选了两门课——“语文”、“化学”

2、一门课程,有两个学生——“杨洋1”、“杨洋2”

	@Resource(name = "sessionFactory")
private SessionFactory sessionFactory; @Override
@Transactional
public void insert() {
//getCurrentSession()比openSession()安全,注意web.xml中的配置,不然会报错
Session session = sessionFactory.getCurrentSession();
Transaction tran = session.beginTransaction(); // 1、王晓东1选了两门课
Student s1 = new Student();
s1.setName("王晓东1"); Course c1 = new Course();
c1.setName("语文");
Course c2 = new Course();
c2.setName("化学"); s1.getCourses().add(c1);
s1.getCourses().add(c2); session.save(s1); // 2、英语课有两个学生
// Course c3 = new Course();
// c3.setName("英语");
// Student s2 = new Student();
// s2.setName("杨洋1");
// Student s3 = new Student();
// s3.setName("杨洋2");
// c3.getStudents().add(s2);
// c3.getStudents().add(s3);
// session.save(c3); tran.commit();
session.flush();//
session.close();
}

  注意commit后,要flush刷到数据库

我的SessionFactory使用了SSH框架的注解配置 (根据自己的方式来获得Session即可)

SSH框架搭建完整实例参考我的另一篇博客:  http://www.cnblogs.com/Donnnnnn/p/7481357.html

实现效果:

1、一个学生,选了两门课——“语文”、“化学”

(1)控制台显示执行的sql语句

Hibernate: insert into students (name) values (?)
Hibernate: insert into courses (name) values (?)
Hibernate: insert into courses (name) values (?)
Hibernate: insert into students_courses (student_id, course_id) values (?, ?)
Hibernate: insert into students_courses (student_id, course_id) values (?, ?)

可以看到,往学生表、课程表和中间表都插入了数据

(2)数据库

students表:

courses表:

students_courses表:

2、一门课程,有两个学生——“杨洋1”、“杨洋2”

同上

参考别人博客:

https://www.cnblogs.com/whgk/p/6121593.html

Hibernate多对多映射(双向关联)实例详解——真的更多相关文章

  1. hibernate一对一外键双向关联

    关联是类(类的实例)之间的关系,表示有意义和值得关注的连接. 本系列将介绍Hibernate中主要的几种关联映射 Hibernate一对一主键单向关联Hibernate一对一主键双向关联Hiberna ...

  2. hibernate一对一主键双向关联

    关联是类(类的实例)之间的关系,表示有意义和值得关注的连接. 本系列将介绍Hibernate中主要的几种关联映射 Hibernate一对一主键单向关联Hibernate一对一主键双向关联Hiberna ...

  3. hibernate集合映射inverse和cascade详解

    hibernate集合映射inverse和cascade详解   1.到底在哪用cascade="..."? cascade属性并不是多对多关系一定要用的,有了它只是让我们在插入或 ...

  4. 一对一关联查询注解@OneToOne的实例详解

    表的关联查询比较复杂,应用的场景很多,本文根据自己的经验解释@OneToOne注解中的属性在项目中的应用.本打算一篇博客把增删改查写在一起,但是在改的时候遇到了一些问题,感觉挺有意思,所以写下第二篇专 ...

  5. Entity Framework实例详解

    Entity Framework Code First的默认行为是使用一系列约定将POCO类映射到表.然而,有时候,不能也不想遵循这些约定,那就需要重写它们.重写默认约定有两种方式:Data Anno ...

  6. 【eclipse插件开发实战】 Eclipse插件开发5——时间插件Timer开发实例详解

    Eclipse插件开发5--时间插件Timer开发实例详解 这里做的TimeHelper插件设定为在菜单栏.工具栏提供快捷方式,需要在相应地方设置扩展点,最后弹出窗体显示时间. 在上一篇文章里创建好了 ...

  7. C#操作SQLite方法实例详解

    用 C# 访问 SQLite 入门(1) CC++C#SQLiteFirefox  用 C# 访问 SQLite 入门 (1) SQLite 在 VS C# 环境下的开发,网上已经有很多教程.我也是从 ...

  8. 官网实例详解-目录和实例简介-keras学习笔记四

    官网实例详解-目录和实例简介-keras学习笔记四 2018-06-11 10:36:18 wyx100 阅读数 4193更多 分类专栏: 人工智能 python 深度学习 keras   版权声明: ...

  9. 实例详解 Java 死锁与破解死锁

    锁和被保护资源之间的关系 我们把一段需要互斥执行的代码称为临界区.线程在进入临界区之前,首先尝试加锁 lock(),如果成功,则进入临界区,此时我们称这个线程持有锁:否则呢就等待,直到持有锁的线程解锁 ...

随机推荐

  1. python 自动化测试Jenkins 持续集成

    一直在做 python 自动化测试,但是脚本的执行之前是运维来维护的,通过 saltstack, 自己并未做过多的研究,后续可以研究一下 saltstack. 今天先研究一下使用 github 管理项 ...

  2. TCHART类型

    private Steema.TeeChart.Styles.Pie pieSeries1; private Steema.TeeChart.Styles.Pie pieSeries2; privat ...

  3. R quantile函数 | cut函数 | sample函数 | all函数 | scale函数 | do.call函数

    取出一个数字序列中的百分位数 1. 求某一个百分比 x<-rnorm(200) quantile(x,0.9) 2. 求一系列的百分比 quantile(x,c(0.1,0.9)) quanti ...

  4. java---->Itellij idea报错:错误: 找不到或无法加载主类 main

      没有设置好正确的类路径 点击上面圈红色处,在点击Edit Configuration,进入下面设置界面 切换到下面这个界面 红色×消失,运行正常,截图如下

  5. 20170922xlVBA_GetCellTextFromWordDocument

    Sub GetCellTextFromWordDocument() '应用程序设置 Application.ScreenUpdating = False Application.DisplayAler ...

  6. Nginx安装与使用 及在redhat 中的简单安装方式

    首先说下在redhat中的安装方法, 正常安装nginx 需要安装很多的依赖,最后再安装nginx,而且很容易出错. 在nginx官方上有这么一段描述: Pre-Built Packages for ...

  7. GreenDao使用解析

    GreenDao是一个轻量级的数据库框架,相比Xutils 等他的速度和效率会更快更好 这里展示GreenDao 的使用方法 ①建立 compile 'org.greenrobot:greendao: ...

  8. Android 和 JS交互方法初探

    起初有个需求,就是需要监听网页的图片点击,然后图片单独跳转到另一个页面单独显示 这里就需要用JS和Android Native方法之间的通信 先说上面的解决办法之前先引出两个Android的方法 1: ...

  9. python-flask-请求源码流程

    启动先执行manage.py 中的    app.run() class Flask(_PackageBoundObject): def run(self, host=None, port=None, ...

  10. 11. Container With Most Water C++

    知识点:双指针遍历大大减少不必要的比较和计算 方法1:Brute Force(执行时间惨不忍睹,共进行n(n-1)/2次比较) class Solution { public: int maxArea ...