一个学生可以选多门课

一门课程有多个学生上

实现步骤:

一、学生

(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. 在dos输入pybot显示不是内部命令,或者显示chromedriver.exe要加入到path中?

    在dos输入pybot显示不是内部命令,或者显示chromedriver.exe要加入到path中? 一直使用robot framework编写脚本,结果有一天输入 pybot XXXX.robot ...

  2. Java Spring JDBC访问数据库

    一.首先采用org.springframework.jdbc.datasource.DriverManagerDataSource类进行实现 1.applicationContext.xml配置如下: ...

  3. joomla 的语言翻译

    最近装了留言板组件 phocaguestbook,发觉没有中文翻译. 于是在 components\com_phocaguestbook\language\en-GB 找到了英文的文件. 依葫芦画瓢, ...

  4. 算法:最短路径之弗洛伊德(Floyd)算法

    https://cloud.tencent.com/developer/article/1012420 为了能讲明白弗洛伊德(Floyd)算法的主要思想,我们先来看最简单的案例.图7-7-12的左图是 ...

  5. Django用户认证组件

    用户认证 主要分两部分: 1.auth模块   from django.contrib import auth 2.User对象 from django.contrib.auth.models imp ...

  6. php url处理

    http_build_query() $data = array("name"=>"callback" , "value"=>& ...

  7. spring boot 多数据源配置与使用

    在介绍使用JdbcTemplate和Spring-data-jpa时,都使用了单数据源.在单数据源的情况下,Spring Boot的配置非常简单,只需要在application.properties文 ...

  8. HDU-4587-tarjin/割点

    http://acm.hdu.edu.cn/showproblem.php?pid=4587 给出一幅无向图,问除去两个点之后子图的最大联通分量个数. 考虑每次ban一个点然后跑一遍tarjin统计下 ...

  9. oracle数据库中字符乱码

    1.1         88.152 os已安装中文包,以下确认os层面中文是否可以显示 1.2         88.153 os没有安装中文包,以下确认os层面中文无法显示 1.3         ...

  10. flask-系统介绍及环境搭建1

    1.系统介绍 前台首页-电影筛选-电影列表- 播放详情-评论:收藏-搜索-注册-登录-会员中心(修改会员资料,查看评论记录,登录日志,收藏电影). 后台-标签-电影管理-预告-会员-评论-收藏-日志- ...