JPA的一对多,多对多用法
一、@OneToOne关系映射
JPA使用@OneToOne来标注一对一的关系。
实体 People :用户。
实体 Address:家庭住址。
People 和 Address 是一对一的关系。
这里用两种方式描述JPA的一对一关系。
一种是通过外键的方式(一个实体通过外键关联到另一个实体的主键);
另外一种是通过一张关联表来保存两个实体一对一的关系。
1、通过外键的方式
people 表(id,name,sex,birthday,address_id)
address 表(id,phone,zipcode,address)
People.java
- @Entity
- public class People {
- @Id
- @GeneratedValue(strategy = GenerationType.IDENTITY)
- @Column(name = "id", nullable = false)
- private Long id;//id
- @Column(name = "name", nullable = true, length = 20)
- private String name;//姓名
- @Column(name = "sex", nullable = true, length = 1)
- private String sex;//性别
- @Column(name = "birthday", nullable = true)
- private Timestamp birthday;//出生日期
- @OneToOne(cascade=CascadeType.ALL)//People是关系的维护端,当删除 people,会级联删除 address
- @JoinColumn(name = "address_id", referencedColumnName = "id")//people中的address_id字段参考address表中的id字段
- private Address address;//地址
- }
关联的实体的主键一般是用来做外键的。但如果此时不想主键作为外键,则需要设置referencedColumnName属性。当然这里关联实体(Address)的主键 id 是用来做主键,所以这里第20行的 referencedColumnName = "id" 实际可以省略。
Address.java
- @Entity
- public class Address {
- @Id
- @GeneratedValue(strategy = GenerationType.IDENTITY)
- @Column(name = "id", nullable = false)
- private Long id;//id
- @Column(name = "phone", nullable = true, length = 11)
- private String phone;//手机
- @Column(name = "zipcode", nullable = true, length = 6)
- private String zipcode;//邮政编码
- @Column(name = "address", nullable = true, length = 100)
- private String address;//地址
- //如果不需要根据Address级联查询People,可以注释掉
- // @OneToOne(mappedBy = "address", cascade = {CascadeType.MERGE, CascadeType.REFRESH}, optional = false)
- // private People people;
- }
2、通过关联表的方式来保存一对一的关系。
people 表(id,name,sex,birthday)
address 表 (id,phone,zipcode,address)
people_address (people_id,address_id)
只需要创建 People 和 Address 两个实体
People.java
- @Entity
- public class People {
- @Id
- @GeneratedValue(strategy = GenerationType.IDENTITY)
- @Column(name = "id", nullable = false)
- private Long id;//id
- @Column(name = "name", nullable = true, length = 20)
- private String name;//姓名
- @Column(name = "sex", nullable = true, length = 1)
- private String sex;//性别
- @Column(name = "birthday", nullable = true)
- private Timestamp birthday;//出生日期
- @OneToOne(cascade=CascadeType.ALL)//People是关系的维护端
- @JoinTable(name = "people_address",
- joinColumns = @JoinColumn(name="people_id"),
- inverseJoinColumns = @JoinColumn(name = "address_id"))//通过关联表保存一对一的关系
- private Address address;//地址
- }
Address.java
不变
二、@OneToMany 和 @ManyToOne
实体 Author:作者。
实体 Article:文章。
Author 和 Article 是一对多关系(双向)。那么在JPA中,如何表示一对多的双向关联呢?
JPA使用@OneToMany和@ManyToOne来标识一对多的双向关联。一端(Author)使用@OneToMany,多端(Article)使用@ManyToOne。
在JPA规范中,一对多的双向关系由多端(Article)来维护。就是说多端(Article)为关系维护端,负责关系的增删改查。一端(Author)则为关系被维护端,不能维护关系。
一端(Author)使用@OneToMany注释的mappedBy="author"属性表明Author是关系被维护端。
多端(Article)使用@ManyToOne和@JoinColumn来注释属性 author,@ManyToOne表明Article是多端,@JoinColumn设置在article表中的关联字段(外键)。
Author.java
- @Entity
- public class Author {
- @Id // 主键
- @GeneratedValue(strategy = GenerationType.IDENTITY) // 自增长策略
- private Long id; //id
- @NotEmpty(message = "姓名不能为空")
- @Size(min=2, max=20)
- @Column(nullable = false, length = 20)
- private String name;//姓名
- @OneToMany(mappedBy = "author",cascade=CascadeType.ALL,fetch=FetchType.LAZY)
- //级联保存、更新、删除、刷新;延迟加载。当删除用户,会级联删除该用户的所有文章
- //拥有mappedBy注解的实体类为关系被维护端
- //mappedBy="author"中的author是Article中的author属性
- private List<Article> articleList;//文章列表
- }
Article.java
- @Entity
- public class Article {
- @Id
- @GeneratedValue(strategy = GenerationType.IDENTITY) // 自增长策略
- @Column(name = "id", nullable = false)
- private Long id;
- @NotEmpty(message = "标题不能为空")
- @Size(min = 2, max = 50)
- @Column(nullable = false, length = 50) // 映射为字段,值不能为空
- private String title;
- @Lob // 大对象,映射 MySQL 的 Long Text 类型
- @Basic(fetch = FetchType.LAZY) // 懒加载
- @NotEmpty(message = "内容不能为空")
- @Size(min = 2)
- @Column(nullable = false) // 映射为字段,值不能为空
- private String content;//文章全文内容
- @ManyToOne(cascade={CascadeType.MERGE,CascadeType.REFRESH},optional=false)//可选属性optional=false,表示author不能为空。删除文章,不影响用户
- @JoinColumn(name="author_id")//设置在article表中的关联字段(外键)
- private Author author;//所属作者
- }
最终生成的表结构
article 表(id,title,conten,author_id)
author 表(id,name)
三、多对多 @ManyToMany
实体 User:用户。
实体 Authority:权限。
用户和权限是多对多的关系。一个用户可以有多个权限,一个权限也可以被很多用户拥有。
JPA中使用@ManyToMany来注解多对多的关系,由一个关联表来维护。这个关联表的表名默认是:主表名+下划线+从表名。(主表是指关系维护端对应的表,从表指关系被维护端对应的表)。这个关联表只有两个外键字段,分别指向主表ID和从表ID。字段的名称默认为:主表名+下划线+主表中的主键列名,从表名+下划线+从表中的主键列名。
需要注意的:
1、多对多关系中一般不设置级联保存、级联删除、级联更新等操作。
2、可以随意指定一方为关系维护端,在这个例子中,我指定 User 为关系维护端,所以生成的关联表名称为: user_authority,关联表的字段为:user_id 和 authority_id。
3、多对多关系的绑定由关系维护端来完成,即由 User.setAuthorities(authorities) 来绑定多对多的关系。关系被维护端不能绑定关系,即Game不能绑定关系。
4、多对多关系的解除由关系维护端来完成,即由Player.getGames().remove(game)来解除多对多的关系。关系被维护端不能解除关系,即Game不能解除关系。
5、如果 User 和 Authority 已经绑定了多对多的关系,那么不能直接删除 Authority,需要由 User 解除关系后,才能删除 Authority。但是可以直接删除 User,因为 User 是关系维护端,删除 User 时,会先解除 User 和 Authority 的关系,再删除 Authority。
User.java
- @Entity
- public class User {
- @Id
- @GeneratedValue(strategy = GenerationType.IDENTITY)
- private Long id;
- @NotEmpty(message = "账号不能为空")
- @Size(min=3, max=20)
- @Column(nullable = false, length = 20, unique = true)
- private String username; // 用户账号,用户登录时的唯一标识
- @NotEmpty(message = "密码不能为空")
- @Size(max=100)
- @Column(length = 100)
- private String password; // 登录时密码
- @ManyToMany
- @JoinTable(name = "user_authority",joinColumns = @JoinColumn(name = "user_id"),
- inverseJoinColumns = @JoinColumn(name = "authority_id"))
- //1、关系维护端,负责多对多关系的绑定和解除
- //2、@JoinTable注解的name属性指定关联表的名字,joinColumns指定外键的名字,关联到关系维护端(User)
- //3、inverseJoinColumns指定外键的名字,要关联的关系被维护端(Authority)
- //4、其实可以不使用@JoinTable注解,默认生成的关联表名称为主表表名+下划线+从表表名,
- //即表名为user_authority
- //关联到主表的外键名:主表名+下划线+主表中的主键列名,即user_id
- //关联到从表的外键名:主表中用于关联的属性名+下划线+从表的主键列名,即authority_id
- //主表就是关系维护端对应的表,从表就是关系被维护端对应的表
- private List<Authority> authorityList;
- }
注意:如注释中所言,上面的第20-21行的@JoinTable可以省略,默认可以生成
Authority.java
- @Entity
- public class Authority {
- @Id
- @GeneratedValue(strategy = GenerationType.IDENTITY)
- private Integer id;
- @Column(nullable = false)
- private String name; //权限名
- @ManyToMany(mappedBy = "authorityList")
- private List<User> userList;
- }
测试 添加
- @SpringBootTest
- @RunWith(SpringRunner.class)
- public class UserRepositoryTest {
- @Autowired
- private UserRepository userRepository;
- @Autowired
- private AuthorityRepository authorityRepository;
- @Test
- public void saveAuthority() {
- Authority authority = new Authority();
- authority.setId(1);
- authority.setName("ROLE_ADMIN");
- authorityRepository.save(authority);
- }
- @Test
- public void saveUser() {
- User user = new User();
- user.setUsername("admin");
- user.setPassword("123456");
- Authority authority = authorityRepository.findById(1).get();
- List<Authority> authorityList = new ArrayList<>();
- authorityList.add(authority);
- user.setAuthorList(authorityList);
- userRepository.save(user);
- }
- }
先运行 saveAuthority 添加一条权限记录,
然后运行 saveUser 添加一条用户记录,与此同时,user_authority 表中也自动插入了一条记录
测试 删除
删除用户
- @SpringBootTest
- @RunWith(SpringRunner.class)
- public class UserRepositoryTest {
- @Autowired
- private UserRepository userRepository;
- @Test
- public void deleteUser() {
- userRepository.deleteById(1L);
- }
- }
user 表中删除一条记录,同时 user_authority 能够级联删除一条记录
参考:http://www.cnblogs.com/luxh/archive/2012/05/30/2527123.html
再次更新
其中 @OneToMany 和 @ManyToOne 用得最多,这里再补充一下
关于级联,一定要注意,要在关系的维护端,即 One 端。
比如 作者和文章,作者是One,文章是Many;文章和评论,文章是One,评论是Many。
cascade = CascadeType.ALL 只能写在 One 端,只有One端改变Many端,不准Many端改变One端。
特别是删除,因为 ALL 里包括更新,删除。
如果删除一条评论,就把文章删了,那算谁的。所以,在使用的时候要小心。一定要在 One 端使用。
举例
One 端
Many 端
JPA的一对多,多对多用法的更多相关文章
- JPA 一对一 一对多 多对一 多对多配置
1 JPA概述 1.1 JPA是什么 JPA (Java Persistence API) Java持久化API.是一套Sun公司 Java官方制定的ORM 方案,是规范,是标准 ,sun公司自己并没 ...
- JPA实体关系映射:@ManyToMany多对多关系、@OneToMany@ManyToOne一对多多对一关系和@OneToOne的深度实例解析
JPA实体关系映射:@ManyToMany多对多关系.@OneToMany@ManyToOne一对多多对一关系和@OneToOne的深度实例解析 今天程序中遇到的错误一 org.hibernate.A ...
- JPA级联(一对一 一对多 多对多)注解【实际项目中摘取的】并非自己实际应用
下面把项目中的用户类中有个:一对一 一对多 多对多的注解对应关系列取出来用于学习 说明:项目运行正常 问题类:一对多.一对一.多对多 ============一对多 一方的设置 @One ...
- day 69-70 一对一 一对多 多对一联表查询
day 69 orm操作之表关系,多对多,多对一 多对一/一对多, 多对多{类中的定义方法} day69 1. 昨日内容回顾 1. 单表增删改查 2. 单表查询API 返回QuerySet对象的: 1 ...
- JPA中实现双向多对多的关联关系(附代码下载)
场景 JPA入门简介与搭建HelloWorld(附代码下载): https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/103473937 ...
- SSAS中事实表中的数据如果因为一对多或多对多关系复制了多份,在维度上聚合的时候还是只算一份
SSAS事实表中的数据,有时候会因为一对多或多对多关系发生复制变成多份,如下图所示: 图1 我们可以从上面图片中看到,在这个例子中,有三个事实表Fact_People_Money(此表用字段Money ...
- 使用NHibernate(7)-- 一对一 && 一对多 && 多对多
1, 一对一. 对于数据量比较大的时候,考虑查询的性能,肯能会把一个对象的属性分到两个表中存放:比如用户和用户资料,经常使用的一般是Id和用户名,用户资料(学校,籍贯等)是不经常被查询的,所以就会分成 ...
- Hibernate 集合映射 一对多多对一 inverse属性 + cascade级联属性 多对多 一对一 关系映射
1 . 集合映射 需求:购物商城,用户有多个地址. // javabean设计 // javabean设计 public class User { private int userId; privat ...
- mybatis的执行流程 #{}和${} Mysql自增主键返回 resultMap 一对多 多对一配置
n Mybatis配置 全局配置文件SqlMapConfig.xml,配置了Mybatis的运行环境等信息. Mapper.xml文件即Sql映射文件,文件中配置了操作数据库的Sql语句.此文件需要在 ...
- 2018.11.4 Hibernate中一对、多对多的关系
简单总结一下 多表关系 一对多/多对一 O 对象 一的一方使用集合. 多的一方直接引用一的一方. R 关系型数据库 多的一方使用外键引用一的一方主键. M 映射文件 一: 多: 操作: 操作管理级别属 ...
随机推荐
- ffmpeg 详解
来源:http://blog.itpub.net/9399028/viewspace-1242300/ FFMPEG详解 认识FFMPEG FFMPEG堪称自由软件中最完备的一套多媒体支持库,它几 ...
- Spring中获取外部配置文件中的属性值
很多时候需要将配置信息从程序中剥离粗来,Spring现在提供的方法是通过@Value注解和<context:placeholder>来获取配置文件中的配置信息.这里给出一个简单的例子. 首 ...
- Linux基础:时间同步工具Chrony
在Linux下,默认情况下,系统时间和硬件时间,并不会自动同步.在Linux运行过程中,系统时间和硬件时间以异步的方式运行,互不干扰.硬件时间的运行,是靠Bios电池来维持,而系统时间,是用CPU t ...
- 【转】学习ARM为什么首选IMAX6??
ARM作为目前嵌入式行业主流的架构,已经让越来越多从事电子行业的朋友了解,并且高校对于嵌入式的学习,很多直接从ARM开始,目前ARM的嵌入式培训也越来越多,足以说明现在嵌入式行业有多火.目前主流的AR ...
- Windows & Ubuntu 双系统完美卸载Ubuntu(不残留,无污染)
双系统卸载Ubuntu时,如若直接从Windows磁盘管理里格式化Ubuntu分区,由于Ubuntu的引导盘的原因,会导致电脑启动时出现问题,所以不建议这样的操作. 卸载Ubuntu前需要区分BIOS ...
- django项目中使用邮箱找回密码功能
本文使用qq邮箱,需要登录邮箱,在设置-账户里面开启SMTP服务,要记下授权码 前端html {#找回密码的表单#} <form action="" method=" ...
- 警告:Establishing SSL connection without server’s identity verification is not recommended
SpringBoot启东时红色警告: Mon Jun 04 00:53:48 CST 2018 WARN: Establishing SSL connection without server's i ...
- 微信开发:"errcode": -1000,"errmsg": "system error"错误的解决办法
最近在微信开发使用微信公众平台接口调试工具时遇到错误. 错误再现:使用appid及appsecret在该测试工具中获取access_token,检查问题时,校验全部通过,但是无法获取access_to ...
- python预课03 三元表达式示例,函数定义示例,七段彩码管绘制示例
三元表达式 s = '不下雨' if s == '下雨': print('带伞') if s == '不下雨': print('不带伞') #等效与以下语句 print('带伞' if s == '下 ...
- js地理定位对象 navigator.geolocation
浏览器信息的存储对象(navigator): 在新的API标准中,可以通过navigator.geolocation来获取设备的当前位置,返回一个位置对象,用户可以从这个对象中得到一些经纬度的相关信息 ...