测试开发专题:spring-boot如何使用JPA进行双向一对多配置
本片文章我们主要介绍spring-boot如何进行JPA的配置以及如何进行实体间的一对多配置。
依赖准备
要在spring-boot使用jpa需要在项目中有进入相关的依赖,pom文件里加入下面内容
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<scope>runtime</scope>
</dependency>
项目的配置文件中需要对数据库链接以及jpa进行配置:
spring:
datasource:
url: jdbc:mysql://localhost:3306/missyou?characterEncoding=utf-8&serverTimezone=GMT%2B8
username: root
password: 12345678
jpa:
hibernate:
ddl-auto: update # 只针对新增的entity创建表
properties:
hibernate:
show_sql: true # 在对数据库进行操作的时候打印出sql,方便在生产环境排查问题
format_sql: true # 打印sql的时候进行格式化,看起来方便
jpa实战
- 一对多关系
先定义两个实体类Banner和BannerItem,一个Banner可以对应多个BannerItem,属于典型的一对多的关系
@Entity
public class Banner {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private long id;
@Column(length = 16)
private String name;
@Transient // 表明这个字段不会映射到表中的字段
private String description;
private String img;
private String title;
@OneToMany
private List<BannerItem> items; // 关联属性,导航属性
}
@Entity
public class BannerItem {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String img;
private String keyword;
private Short type;
private String name;
}
先对上面代码中的一些注解进行说明:
@Entity标记当前类为一个实体,对应数据库中的一张表,用来表示这张表信息,类名默认就是表名,jpa会根据命名规则自动小写并加下划线,比如BannerItem实体,生成的表名就是banner_item
@Id表名标记的字段将作为主键
@GeneratedValue(strategy = GenerationType.IDENTITY) 设置id为自增长
@Column(length = 16) 设置字段的属性,比如长度、是否为空、是否唯一等
@OneToMany可以使用两个实体建立一对多的关系,也就是一个Banner可以包含多个BannerItem
@Transient注解表示被标记的当前字段不会映射到数据库表的字段,也就是说生成的表中不会包含这个字段
此时运行程序,就会在数据库中创建对应的表:

这里生成了三张表,banner表和banner_item表我们知道是对应的Banner实体和BannerItem实体,那banner_items这张表是怎么回事呢?
在解释这张表之前,我们先来看一下这个表的结构:

表里只有两个字段,banner_id和items_id,分别是banner的id和banner_item的id,这就是说这张表相当于一张关系表,在多对多的时候我们才需要一张关系表,而现在只是一对多,为什么会产生这张表呢?
这是因为在上面的实体定义的时候,我们只是给这两个实体建立了关系,并没有任何一个字段来表明哪一个BannerItem属于哪一个Banner,那JPA看到你没有做,那就我来做,他会自动帮你创建这样一张关系表来维护这种归属关系。那有没有办法去掉这张表呢,当然有。
这个时候就需要新增一个外键字段了,来表明一个banner_item到底是归属于哪一个banner,我们来修改一下上面的实体定义:
Banner实体里做如下修改:
@OneToMany
@JoinColumn(name = "bannerId")
private List<BannerItem> items;
@JoinColumn注解指定外键
BannerItem实体增加一个字段:bannerId
private Long bannerId;
删除之前的表,重新运行程序

这个时候生成的表就只有两个了,在banner_item表里会新增一个外键字段。
下面我们通过一个程序实例来验证上面的内容:
1、构造数据
表结构已经构造好了,现在我们填充一些数据,来使用JPA进行数据查询
INSERT INTO missyou.banner (id, img, name, title) VALUES (1, 'http://sss.jpg', '顶部banner', '顶部banner');
INSERT INTO missyou.banner (id, img, name, title) VALUES (2, 'http://aaa.jpg', '顶部banner2', '顶部banner2');
INSERT INTO missyou.banner_item (id, banner_id, img, keyword, name, type) VALUES (1, 1, 'http://www.jpj', '衣服', '阿迪促销', 1);
INSERT INTO missyou.banner_item (id, banner_id, img, keyword, name, type) VALUES (2, 1, 'http://www.jpj', '鞋', '阿迪促销', 1);
2、新建Controller\service\repository
BannerRepository仓储层,主要负责读取数据库
@Repository
public interface BannerRepository extends JpaRepository<Banner, Long> {
Banner findOneById(Long id);
Banner findOneByName(String name);
}
sevice层,具体的业务逻辑
@Service
public class BannerServiceImpl implements BannerService{
@Autowired
private BannerRepository bannerRepository;
@Override
public Banner getByName(String name) {
return bannerRepository.findOneByName(name);
}
}
controller层,负责接收客户端请求返回数据
@GetMapping("/banner/name/{name}")
public Banner getBannerByName(@PathVariable String name){
return bannerService.getByName(name);
}
请求接口:

可以看到数据已经查询出来了。
- 双向一对多
上面的示例是根据banner_id去查关联的一组banner_item,但是有些场景我们可能需要通过某个item_id去查是属于哪个banner,那这就是一种的反向的一对多,接下来我们看一下这总反向的一对多如何配置。
我们之前配置了一个items属性
@OneToMany
@JoinColumn(name = "bannerId")
private List<BannerItem> items;
这个属性就可以帮我们定位到某个banenr关联的一组bannerItem,可以叫做导航属性,那么反过来,如何从一个item定位到一个Banner,怎么设置导航属性?
首先需要在BannerItem实体类里面增加一个banner属性,并使用如下注解进行标记
@ManyToOne
@JoinColumn(name = "bannerId")
private Banner banner;
修改Banner实体
@OneToMany(mappedBy = "banner")
private List<BannerItem> items;
这里其实就是把@JoinColumn(name = "bannerId")移动到BannerItem里面。并且在BanenrItem里要删除掉之前定义的外键字段bannerId。
这是因为在设置双向一对多的关系时,会默认在banner_item表里生成一个外键,如果自己再写一个就会造成重复,程序就会报错。
那如果自己很想指定,也可以,进行如下配置
@ManyToOne
@JoinColumn(insertable = false, updatable = false, name = "bannerId")
private Banner banner;
总结
以上就是我们介绍的关于在spring-boot中如何进行实体间的单向一对多和双向一对多的配置,关于要在实际项目如何使用哪一种方式,还是要结合自己的业务,一般不需要进行双向的配置,而且一般都不会去设置物理外键,后面我们再讨论如何进行多对多的配置以及如何去除物理外键。
欢迎大家去 我的博客 瞅瞅,里面有更多关于测试实战的内容哦!!
测试开发专题:spring-boot如何使用JPA进行双向一对多配置的更多相关文章
- 初识在Spring Boot中使用JPA
前面关于Spring Boot的文章已经介绍了很多了,但是一直都没有涉及到数据库的操作问题,数据库操作当然也是我们在开发中无法回避的问题,那么今天我们就来看看Spring Boot给我们提供了哪些疯狂 ...
- 【实验一 】Spring Boot 集成 hibernate & JPA
转眼间,2018年的十二分之一都快过完了,忙于各类事情,博客也都快一个月没更新了.今天我们继续来学习Springboot对象持久化. 首先JPA是Java持久化API,定义了一系列对象持久化的标准,而 ...
- 从零开始开发一个Spring Boot Starter
一.Spring Boot Starter简介 Starter是Spring Boot中的一个非常重要的概念,Starter相当于模块,它能将模块所需的依赖整合起来并对模块内的Bean根据环境( 条件 ...
- 使用spring boot中的JPA操作数据库
前言 Spring boot中的JPA 使用的同学都会感觉到他的强大,简直就是神器一般,通俗的说,根本不需要你写sql,这就帮你节省了很多时间,那么下面我们来一起来体验下这款神器吧. 一.在pom中添 ...
- 快速开发架构Spring Boot 从入门到精通 附源码
导读 篇幅较长,干货十足,阅读需花费点时间.珍惜原创,转载请注明出处,谢谢! Spring Boot基础 Spring Boot简介 Spring Boot是由Pivotal团队提供的全新框架,其设计 ...
- 基于Spring Boot,使用JPA动态调用Sql查询数据
在<基于Spring Boot,使用JPA操作Sql Server数据库完成CRUD>,<基于Spring Boot,使用JPA调用Sql Server数据库的存储过程并返回记录集合 ...
- 基于Spring Boot,使用JPA调用Sql Server数据库的存储过程并返回记录集合
在上一篇<基于Spring Boot,使用JPA操作Sql Server数据库完成CRUD>中完成了使用JPA对实体数据的CRUD操作. 那么,有些情况,会把一些查询语句写在存储过程中,由 ...
- 使用Eclipse开发学习 Spring Boot 教程的内容小结
spring-tool-suite使用教程,并创建spring配置文件 Spring Boot基础教程1-Spring Tool Suite工具的安装 Spring Boot基础教程2-RESTful ...
- 测试开发专题:spring-boot自定义异常返回
上文测试开发专题:spring-boot统一异常捕获我们讨论了java异常以及如何使用Spring-Boot捕获异常,但是没有去说捕获异常后该如何进一步处理,这篇文章我们将对这个遗留的问题进行讨论. ...
随机推荐
- RequestDispatcher.forward() 方法和HttpServletResponse.sendRedirect()方法的区别
RequestDispatcher.forward() 方法和HttpServletResponse.sendRedirect()方法的区别 先贴一段代码 public void logon(Http ...
- 对话Roadstar投资人:一家自动驾驶公司之死(三)
...
11. Roadstar 如何收场? 雷锋网:你觉得 Roadstar 造成今天这样的局面,是什么导致的? 投资人代表 1:刚才我们也数次表达了,在每个人身上,可能每个人的诉求,不能达到同步,与公司的 ...
- 内蒙古特检院利用物联网/RFID技术提高电梯检测水平
随着电梯检验工作信息化进程的进一步深入,内蒙古特检院从检验工作中寻找新方法.新手段,为检验员新引入电梯检验手持终端设备,力求提高电梯检验水平,将"电梯安全惠民工程"落到实处. 电梯 ...
- POJ3614防晒霜 这个贪心有点东西(贪心+优先队列)
这个题是说有C头牛去晒太阳,带了L瓶防晒霜,每瓶防晒霜都有一个SPF值(每瓶防晒霜都能解决一个最短路 ) 每头牛给出了他可以接受防晒霜的上限,和下限,每种防晒霜都给出了SPF值与数量. 从防晒霜的sp ...
- 题解 CF160B 【Unlucky Ticket】
本文为UserUnknown原创 思路 这道题应该怎么做? 可以把输入的数字逐位拆分后存入数组,就像这样存进去: int a[N],b[N] tmp=n; k=1; while(--tmp){ a[k ...
- RF(scalar/list/dict变量)
一.scalar 变量 ${} 定义 scalar 变量 ${} 还用来取值 1.set variable 设置变量 ${name} Set Variable zhangsan log ${nam ...
- Vxlan L3
拓扑图: CE1 <CE1>display current-configuration !Software Version V800R013C00SPC560B560 !Last conf ...
- 题目分享T
题意:蛐蛐国里现在共有n只蚯蚓(n为正整数).每只蚯蚓拥有长度,我们设第i只蚯蚓的长度为a_i(i=1,2,...,n),并保证所有的长度都是非负整数(即:可 能存在长度为0的蚯蚓).每一秒,神刀手会 ...
- vue2.0那些坑之使用elementUI后v-on:click事件不生效问题
最近在维护vue2.0的项目,遇到了不少坑,在这里说下引用elementui之后,使用v-on:click绑定点击事件无效的情况,如下图: 我想阻止冒泡事件,发现无效.这里将@click换成了@cli ...
- C:简单实现BaseCode64编码
What is Base64? 前言 目前来看遇到过Base 16.Base 32.Base 64的编解码,这种编码格式是二进制和文本编码转化,是对称并且可逆的转化.Base 64总共有64个ASCI ...