Hibernate一对多OnetoMany
------------------------Hibernate一对多OnetoMany
要点:
配置在一端。
1.如果是单向关联,即只在一端配置OneToMany,多端不配置ManyToOne。则会创建一张关系表。
2.如果在单向关联的OneToMany上加上@JoinColumn,则会在多端增加一个外键维护关系,而不单独创建关系表。
3.可以配置级联操作 @OneToMany(cascade=CascadeType.ALL),保存一端的时候保存多端
4.如果是双向关联,即在一端配置OneToMany,多端配置ManyToOne,则在多端增加外键字段维护关系,
并且要使用mappedBy或者是inverse确保只有一端维护关系 2.单向关联
@Entity
public class User { @Id
@GeneratedValue
private long id; @Column(name="name")
private String name; @OneToMany(cascade=CascadeType.ALL)
@JoinColumn
private Set<Role> roles = new HashSet<Role>(); } @Entity
public class Role {
@Id
@GeneratedValue
private long id;
@Column
private String name;
} 执行:
User user = new User();
user.setName("zhangsan"); Role r1= new Role();
r1.setName("ADMIN");
Role r2= new Role();
r2.setName("EDITOR"); user.getRoles().add(r1);
user.getRoles().add(r2); session.save(user); 结果:
Hibernate:
insert
into
User
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, id)
values
(?, ?)
Hibernate:
update
Role
set
roles_id=?
where
id=?
Hibernate:
update
Role
set
roles_id=?
where
id=?
保存user后保存两个role 然后更新外键。多两条更新语句。 执行:保存user后删除一个role对象。
User user = new User();
user.setName("zhangsan"); Role r1= new Role();
r1.setName("ADMIN");
Role r2= new Role();
r2.setName("EDITOR"); user.getRoles().add(r1);
user.getRoles().add(r2); session.save(user);
user.getRoles().remove(r1);
结果:
Hibernate:
insert
into
User
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, id)
values
(?, ?)
Hibernate:
update
Role
set
roles_id=?
where
id=? 保存user后保存两个role,然后更新被删除role的外键为Null。此时并没有删除role这个数据。
只需要增加orphanRemoval=true,则会删除已经没有关联关系的子实体
@Entity
public class User { @Id
@GeneratedValue
private long id; @Column(name="name")
private String name; @OneToMany(cascade=CascadeType.ALL,orphanRemoval=true)
@JoinColumn
private Set<Role> roles = new HashSet<Role>();
} 结果:
Hibernate:
insert
into
User
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, id)
values
(?, ?)
Hibernate:
update
Role
set
roles_id=?
where
id=?
Hibernate:
delete
from
Role
where
id=? 将移除的子实体更新外键为null,然后执行删除,效率一般。 4.双向关联
要点:关系必须双向都设置,业务代码中关系也必须双向设置
mappedBy指向多端实体里的一端变量名
inverse和mappedBy是同一个东西,只是inverse是用于xml配置,而mappedBy则是用于注解中。
只有OneToOne,OneToMany,ManyToMany上才有mappedBy属性,ManyToOne不存在该属性; @Entity
public class User { @Id
@GeneratedValue
private long id; @Column(name="name")
private String name; @OneToMany(cascade=CascadeType.ALL,mappedBy="user")//由多端维护关系
private Set<Role> roles = new HashSet<Role>();
} @Entity
public class Role {
@Id
@GeneratedValue
private long id;
@Column
private String name;
@ManyToOne
private User user;
} -------------------mappedBy 只保存一端
T1
执行:
User user = new User();
user.setName("zhangsan"); Role r1= new Role();
r1.setName("ADMIN");
Role r2= new Role();
r2.setName("EDITOR");
//业务代码中双向设置关系
user.getRoles().add(r1);
user.getRoles().add(r2);
r1.setUser(user);
r2.setUser(user); session.save(user);
结果:
Hibernate:
insert
into
User
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, user_id, id)
values
(?, ?, ?)
Hibernate:
insert
into
Role
(name, user_id, id)
values
(?, ?, ?)
执行3条语句,插入结果正确。将保存操作传递到多端了,然后进行了保存。 --------------------------------mappedBy 先保存多端,再保存一端
User user = new User();
user.setName("zhangsan"); Role r1= new Role();
r1.setName("ADMIN");
Role r2= new Role();
r2.setName("EDITOR"); user.getRoles().add(r1);
user.getRoles().add(r2);
r1.setUser(user);
r2.setUser(user);
session.save(r1);
session.save(r2);
session.save(user);//无此句则报错,因为user为瞬时态,而role端无级联操作
结果:
Hibernate:
insert
into
Role
(name, user_id, id)
values
(?, ?, ?)
Hibernate:
insert
into
Role
(name, user_id, id)
values
(?, ?, ?)
Hibernate:
insert
into
User
(name, id)
values
(?, ?)
Hibernate:
update
Role
set
name=?,
user_id=?
where
id=?
Hibernate:
update
Role
set
name=?,
user_id=?
where
id=?
先保存两个role,然后保存user。role负责关系维护,又更新了关系外键。 ---------------------mappedBy 只保存一端,并删除一端集合的一个数据 执行:
User user = new User();
user.setName("zhangsan"); Role r1= new Role();
r1.setName("ADMIN");
Role r2= new Role();
r2.setName("EDITOR");
//业务代码中双向设置关系
user.getRoles().add(r1);
user.getRoles().add(r2);
r1.setUser(user);
r2.setUser(user); session.save(user);
user.getRoles().remove(r1); 结果:
Hibernate:
insert
into
User
(name, id)
values
(?, ?)
Hibernate:
insert
into
Role
(name, user_id, id)
values
(?, ?, ?)
Hibernate:
insert
into
Role
(name, user_id, id)
values
(?, ?, ?) 没有进行删除操作。User不负责关系维护,操作失败。
当指定orphanRemoval=true时删除成功! ---------级联删除
--------------------直接删除一端(数据为T1操作后的数据)
User user = session.get(User.class, 170l);
session.remove(user); 结果:
Hibernate:
delete
from
Role
where
id=?
Hibernate:
delete
from
Role
where
id=?
Hibernate:
delete
from
User
where
id=?
删除了一端和所有多端。删除进行了级联传递,先删除多端的两个Role,然后删除用户。
此时确先删除了多端,然后删除一端?直接先删除一端是不可能的
--------------------直接删除多端
Role r = session.get(Role.class, 175l);
session.remove(r);
结果:
Hibernate:
delete
from
Role
where
id=?
执行成功。相当于执行了mantoone的删除
//删除
User user = session.get(User.class, 167l);
Role role = session.get(Role.class, 168l);
System.out.println(user.getName());
System.out.println(role.getName());
user.getRoles().remove(role);//当载入内存后,必须解除关系,再删除
session.remove(role);//删除成功
Hibernate一对多OnetoMany的更多相关文章
- 【Jpa hibernate】一对多@OneToMany,多对一@ManyToOne的使用
项目中使用实体之间存在一对多@OneToMany,多对一@ManyToOne的映射关系,怎么设置呢? GitHub地址:https://github.com/AngelSXD/myagenorderd ...
- hibernate 一对多双向关联 详解
一.解析: 1. 一对多双向关联也就是说,在加载班级时,能够知道这个班级所有的学生. 同时,在加载学生时,也能够知道这个学生所在的班级. 2.我们知道,一对多关联映射和多对一关联映射是一样的,都是在 ...
- Hibernate一对多单向关联和双向关联映射方法及其优缺点 (待续)
一对多关联映射和多对一关联映射实现的基本原理都是一样的,既是在多的一端加入一个外键指向一的一端外键,而主要的区别就是维护端不同.它们的区别在于维护的关系不同: 一对多关联映射是指在加载一的一端数据的同 ...
- Hibernate一对多操作
--------------------siwuxie095 Hibernate 一对多操作 以客户和联系人为例,客户是一,联系人是多 即 一个客户里面有多个联系人,一个联系人只能属于一个客户 注意: ...
- hibernate 一对多(级联关系)
hibernate 核心配置文件 <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hiber ...
- Java进阶知识10 Hibernate一对多_多对一双向关联(Annotation+XML实现)
本文知识点(目录): 1.Annotation 注解版(只是测试建表) 2.XML版 的实现(只是测试建表) 3.附录(Annotation 注解版CRUD操作)[注解版有个问题:插入值时 ...
- Java进阶知识09 Hibernate一对多单向关联(Annotation+XML实现)
1.Annotation 注解版 1.1.在一的一方加Set 1.2.创建Customer类和Order类 package com.shore.model; import java.util.Hash ...
- Hibernate一对多配置
刚刚学习了Hibernate框架的基础知识,下面我来说说关于Hibernate一对多的配置 首先是大配置 连接数据库 用户名 和密码 能和小配置连接 部门小配置: 员工小配置: 部门实体类 员工实体类 ...
- JPA总结——实体关系映射(一对多@OneToMany)
JPA总结——实体关系映射(一对多@OneToMany) 注意:本文出自“阿飞”的博客,如果要转载本文章,请与作者联系! 并注明来源: http://blog.sina.com.cn/s/blog_4 ...
随机推荐
- 错误:Bean property 'sessionFactory' is not writable or has an invalid setter method.
Caused by: org.springframework.beans.NotWritablePropertyException: Invalid property 'sessionFactory' ...
- .net core webapi带权限的文件下载方法
众所周知,在webapi中,如果有个接口需要权限,一般会将带权限的字段塞进header中.但是,在带权限的文档下载接口中,无论是用post,还是get方式,我们无法设置header头信息.苦恼呀?别急 ...
- Linux3.10.0块IO子系统流程(5)-- 为SCSI命令准备聚散列表
SCSI数据缓冲区组织成聚散列表的形式.Linux内核中表示聚散列表的基本数据结构是scatterlist,虽然名字中有list,但它只对应一个内存缓冲区,聚散列表就是多个scatterlist的组合 ...
- 特殊场景Sql优化
一.大表的大数据量修改 问题: 1.大量的行级锁,长时间阻塞 2.主从延时,大批数据不一致 解决方法: 分批次修改 二.大表的表结构修改 问题:长时间锁表 解决方法: 1.从库修改,主从切换,主库 ...
- SQL-49 针对库中的所有表生成select count(*)对应的SQL语句
题目描述 针对库中的所有表生成select count(*)对应的SQL语句CREATE TABLE `employees` (`emp_no` int(11) NOT NULL,`birth_dat ...
- cocoa pods自己的笔记
备注:这里只是个人的观点,有的地方也是copy,多多指教,个人笔记,有侵犯你们版权的地方还望海涵!!! 卡主不动 安装流程:http://www.tuicool.com/articles/qaMfuy ...
- 1093 字符串A+B
给定两个字符串 A 和 B,本题要求你输出 A+B,即两个字符串的并集.要求先输出 A,再输出 B,但重复的字符必须被剔除. 输入格式: 输入在两行中分别给出 A 和 B,均为长度不超过 106 ...
- LNMP(一)
第二十课LNMP(一) 目录 一.LNMP架构介绍 二.MySQL安装 三.PHP安装 四.Nginx介绍 五.Nginx安装 六.扩展 一.LNMP架构介绍 之前已经学习过LAMP架构,与LAMP相 ...
- STL next_permutation 算法原理和实现
转载自:https://www.cnblogs.com/luruiyuan/p/5914909.html 目标 STL中的next_permutation 函数和 prev_permutation 两 ...
- python自学第12天 模块定义,导入,内置模块
1.定义模块:用来从逻辑上组织python代码(实现一个功能),本质是.py结尾的python 包:本质就是一个目录(必须带有一个_init_.py文件)2.导入方法import module_nam ...