Hibernate 系列教程6-双向多对多
双向多对多inverse配置
如果关系两边都使用默认inverse=false配置,表示关系两边都可以管理中间表的数据
关系其中一边设置inverse=true配置,表示自己放弃管理中间表关系,由对方来管理
如果关系两边都使用默认inverse=true配置,表示关系两边都不管理中间表,是不能这样配置的
hibernate.cfg.xml
<mapping resource="com/jege/hibernate/two/way/manytomany/User.hbm.xml" />
<mapping resource="com/jege/hibernate/two/way/manytomany/Role.hbm.xml" />
User.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.jege.hibernate.two.way.manytomany">
<class name="User" table="t_user">
<id name="id">
<generator class="native"/>
</id>
<property name="name" />
<!-- 配置中间表t_user_role -->
<set name="roles" table="t_user_role">
<!-- 配置当前类User在中间表的主外键名称 -->
<key column="user_id" />
<!-- 多对多 -->
<!-- class必须配置, column配置关联类Role在中间表的主外键名称 -->
<many-to-many class="Role" column="role_id" />
</set>
</class>
</hibernate-mapping>
Role.hbm.xml
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="com.jege.hibernate.two.way.manytomany">
<class name="Role" table="t_role">
<id name="id">
<generator class="native" />
</id>
<property name="name" />
<!-- 配置中间表t_user_role -->
<set name="users" table="t_user_role">
<!-- 配置当前类Role在中间表的主外键名称 -->
<key column="role_id" />
<!-- 多对多 -->
<!-- class必须配置, column配置关联类User在中间表的主外键名称 -->
<many-to-many class="User" column="user_id" />
</set>
</class>
</hibernate-mapping>
Role
private Long id;
private String name;
private Set<User> users = new HashSet<User>();
User
private Long id;
private String name;
private Set<Role> roles = new HashSet<Role>();
CrudTest
package com.jege.hibernate.two.way.manytomany;
import org.hibernate.Session;
import org.junit.Before;
import org.junit.Test;
import com.jege.hibernate.two.way.manytomany.Role;
import com.jege.hibernate.two.way.manytomany.User;
import com.jege.hibernate.util.HibernateUtils;
/**
* @author JE哥
* @email 1272434821@qq.com
* @description:多对多的处理
*/
public class CrudTest {
// 保存2个用户,保存3个角色(会出现5条insert语句)
// 保存中间表:建立用户到角色关系jegeUser1(jegeRole1,jegeRole2),jegeUser2(jegeRole1,jegeRole2,jegeRole3)(会出现5条insert语句)
// 一般保存的:需要写5个save方法
@Before
public void save() throws Exception {
Session session = HibernateUtils.INSTANCE.getSession();
session.beginTransaction();
// 保存2个用户,保存3个角色(会出现5条insert语句)
User jegeUser1 = new User("jegeUser1");
session.save(jegeUser1);
User jegeUser2 = new User("jegeUser2");
session.save(jegeUser2);
Role jegeRole1 = new Role("jegeRole1");
session.save(jegeRole1);
Role jegeRole2 = new Role("jegeRole2");
session.save(jegeRole2);
Role jegeRole3 = new Role("jegeRole3");
session.save(jegeRole3);// (这里的5个save方法会出现5条insert语句)
// 保存中间表:建立用户到角色关系jegeUser1(jegeRole1,jegeRole2),jegeUser2(jegeRole1,jegeRole2,jegeRole3)
jegeUser1.getRoles().add(jegeRole1);
jegeUser1.getRoles().add(jegeRole2);
jegeUser2.getRoles().add(jegeRole1);
jegeUser2.getRoles().add(jegeRole2);
jegeUser2.getRoles().add(jegeRole3);
jegeUser2.getRoles().add(jegeRole3);
session.getTransaction().commit();// 事务里面,更新脏数据(会出现5条insert语句)
session.close();
}
// 删除jegeUser1(由hibernate自动处理,流程是先删除中间表,在删除jegeUser1)
@Test
public void delete1() throws Exception {
Session session = HibernateUtils.INSTANCE.getSession();
User jegeUser = (User) session.get(User.class, 1L);
session.beginTransaction();
session.delete(jegeUser);
session.getTransaction().commit();
session.close();
}
// 删除jegeUser1的关联的角色(中间表),实现不能删除jegeUser1
@Test
public void delete2() throws Exception {
Session session = HibernateUtils.INSTANCE.getSession();
User jegeUser = (User) session.get(User.class, 1L);
session.beginTransaction();
jegeUser.getRoles().clear();
session.getTransaction().commit();
session.close();
}
// 删除jegeUser1的一个角色,实现不能删除jegeUser1
@Test
public void delete3() throws Exception {
Session session = HibernateUtils.INSTANCE.getSession();
User jegeUser = (User) session.get(User.class, 1L);
Role jegeRole = (Role) session.get(Role.class, 1L);
session.beginTransaction();
jegeUser.getRoles().remove(jegeRole);
session.getTransaction().commit();
session.close();
}
// 修改角色:先删除jegeUser1.jegeRole2,在添加jegeUser1.jegeRole3
// 关系jegeUser1(jegeRole1,jegeRole2)变为关系jegeUser1(jegeRole1,jegeRole3)
@Test
public void update() throws Exception {
Session session = HibernateUtils.INSTANCE.getSession();
User jegeUser = (User) session.get(User.class, 1L);
Role jegeRole2 = (Role) session.get(Role.class, 2L);
Role jegeRole3 = (Role) session.get(Role.class, 3L);
session.beginTransaction();
jegeUser.getRoles().remove(jegeRole2);
jegeUser.getRoles().add(jegeRole3);
session.getTransaction().commit();
session.close();
}
}
源码地址
https://github.com/je-ge/hibernate
如果觉得我的文章对您有帮助,请予以打赏。您的支持将鼓励我继续创作!谢谢!
Hibernate 系列教程6-双向多对多的更多相关文章
- Hibernate 系列教程5-双向多对一
主要讲解inverse和cascade的用法 cascade定义的是关系两端对象到对象的级联关系: 而inverse定义的是关系和对象的级联关系(管理外键的值). inverse 属性默认是false ...
- Hibernate 系列教程4-单向多对一
项目图片 hibernate.cfg.xml <mapping resource="com/jege/hibernate/one/way/manytoone/User.hbm.xml& ...
- JPA 系列教程7-双向多对多
双向多对多的ddl语句 同单向多对多表的ddl语句一致 Student package com.jege.jpa.many2many; import java.util.HashSet; import ...
- JPA 系列教程3-单向多对一
JPA中的@ManyToOne 主要属性 - name(必需): 设定"many"方所包含的"one"方所对应的持久化类的属性名 - column(可选): 设 ...
- Hibernate 系列教程16-二级缓存
pom.xml <dependency> <groupId>org.hibernate</groupId> <artifactId>hibernate- ...
- Hibernate 系列教程9-自关联
自关联:本质还是原来双向一对多,原来要配置两个类,现在全部都配置在一个类里面 Employee public class Employee { private Long id; private Str ...
- Hibernate 系列教程7-双向一对一
双向一对一 一对一主要用在 一个一方需要的信息比较少,比如注册的登录信息 另一个一方存储的信息比较多,比如注册之后用户填写的详细信息 实现方式常用的主要有2种: java模型都是一样,其中一个映射文件 ...
- JPA 系列教程6-单向多对多
JPA中的@ManyToMany @ManyToMany注释表示模型类是多对多关系的一端. @JoinTable 描述了多对多关系的数据表关系. name 属性指定中间表名称 joinColumns ...
- Hibernate 系列教程17-查询缓存
在二级缓存配置成功的基础上进行查询缓存配置 Product public class Product { private Long id; private String name; Product.h ...
随机推荐
- POJ 1323 Game Prediction#贪心
(- ̄▽ ̄)-* //既然是求最少能胜几次 //说明对方是要尽可能让我输 //但为了避免浪费,对方会用比我的牌大的牌中的最小pip的牌来击败我 #include<iostream> #in ...
- Java 集合 散列表hash table
Java 集合 散列表hash table @author ixenos 摘要:hash table用链表数组实现.解决散列表的冲突:开放地址法 和 链地址法(冲突链表方式) hash table 是 ...
- strstr() strpos() 获取db报错,判断报错中是否包含字符串,判断错误类型
model中直接获取添加公司的错误.(公司名称不能重复) $enterprise_id = $this->add($enterprisedata ); $err = $this->getD ...
- urllib2 之info 学习
之前介绍了根据old_url获取真实url的geturl的方法,而根据urlopen返回的应答对象的info方法可以获取服务器发送头部headers的内容,并且通过字典形式反馈出来,同样测试代码如下: ...
- 【Loadrunner】初学Loadrunner——安装
一.准备工作 1.下载Loadrunner可以参考网上百度得到的可以在下面这个地址下载,比较大,4G左右 http://www.genilogix.com/downloads/loadrunner/l ...
- FireFox站点标识按钮
Firefox 的站点标识按钮可以为您提供有关您访问的网站的详细信息.通过站点标识按钮,您可以了解到站点的加密信息.验证信息.网站所有者和网站验证者.这有助于避免恶意网站获得您的重要信息. 站点标识按 ...
- Colorful(Folders星语多彩文件夹) v1.7绿色版
软件名称:星语多彩文件夹ColorfulFolders v1.7绿色版软件类别:国产软件运行环境:WinXP/2003/Vista/Win7/Win2008软件语言:简体中文授权方式:免费版软件大小: ...
- 2016-11-10linux
---恢复内容开始--- 新建用户natasha,uid为88,gid为6,备注信息为"master" 修改natasha用户的家目录为/Natasha 查看用户信息配 ...
- js常用API 数据类型 基本类型,基本包装类型,引用类型 Object String Array Boolean Number Date Math
数据类型 变量.作用域及内存 基础类型(primitive value):Undefined.Null.Boolean.Number和String.这些类型在内存中分别占用固定大小的空间,他们的值保存 ...
- 二十二、oracle pl/sql分类二 函数
函数用于返回特定的数据,当建立函数时,在函数头部必须包含return子句.而在函数体内必须包含return语句返回的数据.我们可以使用create function来建立函数. 1).接下来通过一个案 ...