【持久层框架】- SpringData - JPA

news/2024/4/30 21:01:45/文章来源:https://blog.csdn.net/qq_43843951/article/details/127150515

SpringData - JPA

😄生命不息,写作不止
🔥 继续踏上学习之路,学之分享笔记
👊 总有一天我也能像各位大佬一样
🏆 一个有梦有戏的人 @怒放吧德德
🌝分享学习心得,欢迎指正,大家一起学习成长!

在这里插入图片描述

JPA

  • SpringData - JPA
    • JPA简介
    • 什么是ORM?
    • SpringData-JPA
    • SpringBoot整合JPA
      • 导入依赖
      • 编写配置文件
      • 创建实体类
      • 启动项目
      • 对表操作
        • 增加数据
        • 查看数据
        • 分页查找
        • 根据ID查找
    • 方法命名规则自定义查询
      • 编写Repository
    • 使用JPQL的方式查询(注解式SQL)
      • 计数
      • 参数传递
    • 实体关系对应
      • 一对一
      • 一对多
      • 多对多
    • 文章推荐

JPA简介

JPA 即Java Persistence API。是一款持久层框架,中文名Java持久层API,是JDK 5.0注解或XML描述对象-关系表的映射关系,并将运行期的实体对象持久化到数据库中。JPA的对象关系映射(ORM)模型是基于Hibernate。是一款面向对象的ORM框架,JPA不需要关心数据库字段,通过注解将数据库表与实体完成映射,在java中的操作只需要对实体进行操作即可。

什么是ORM?

ORM(Object-Relational Mapping) 表示对象关系映射。在面向对象的软件开发中,通过ORM,就可以把对象映射到关系型数据库中。只要有一套程序能够做到建立对象与数据库的关联,操作对象就可以直接操作数据库数据,就可以说这套程序实现了ORM对象关系映射。也就是说ORM是建立了一个实体与数据库表的关系,使得开发者对实体的直接操作而不是对数据库的操作,但操作实体也就等同于操作了数据库。
ORM框架还有:MyBatis、Hibernate

SpringData-JPA

Spring Data JPA 是 Spring 基于 ORM 框架、JPA 规范的基础上封装的一套JPA应用框架,可使开发者用极简的代码即可实现对数据库的访问和操作。
它提供了包括增删改查等在内的常用功能,且易于扩展!学习并使用 Spring Data JPA 可以极大提高开发效率!
Spring Data JPA 让我们解脱了DAO层的操作,基本上所有CRUD都可以依赖于它来实现
在实际的工作工程中,推荐使用Spring Data JPA + ORM(如:hibernate)完成操作,这样在切换不同的ORM框架时提供了极大的方便,同时也使数据库层操作更加简单,方便解耦。

SpringBoot整合JPA

接下来通过一个案例简单学习一下JPA,并且在案例中学习一些相关知识。

导入依赖

需要导入jpa坐标和mysql依赖包

<dependencies><!--jpa--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId></dependency><!--Mysql依赖包--><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><scope>runtime</scope></dependency>
</dependencies>

编写配置文件

先配置数据源,采用mysql8.0。还包括配置jpa的相关配置。主要是配置jpa的sql日志打印和自动创建表格。

spring:datasource:driver-class-name: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://localhost:3306/springboot_demo?useUnicode=true&useSSL=false&characterEncoding=utf8&serverTimezone=Asia/Shanghaiusername: rootpassword: 12356
jpa:show-sql: truehibernate:ddl-auto: create # 自动创建表格,之后改成update

jpa的底层是通过hibernate执行的,通过配置ddl-auto能够自动创建表,但是这个不建议使用,如果是create,在项目运行的时候就会把存在的表删除在添加。详情如下:
ddl-auto属性用于设置自动表定义,可以实现自动在数据库中为我们创建一个表,表的结构会根据我们定义的实体类决定,它有4种

  • create 启动时删数据库中的表,然后创建,退出时不删除数据表
  • create-drop 启动时删数据库中的表,然后创建,退出时删除数据表 如果表不存在报错
  • update 如果启动时表格式不一致则更新表,原有数据保留
  • validate 项目启动表结构进行校验 如果不一致则报错

接下来就可以开始体验jpa了

创建实体类

可见通过注解可以实现实体与数据库的直接映射,jpa操作的就是这些实体就等同操作数据库的表结构。

package com.jpa.demo.domain;import lombok.Data;import javax.annotation.Generated;
import javax.persistence.*;
import java.io.Serializable;/*** @author: lyd* @description: 实体User* @Date: 2022/10/2*/
@Data
@Entity
@Table(name = "sys_user")
public class User implements Serializable {@Id@Column(name = "user_id")@GeneratedValue(strategy = GenerationType.IDENTITY) // 自增private Long id;private String username;private String nickName;private String password;}
  • @Data:lombok下的注解,主要是方便开发,减少对get、set、tostring的编写。这个需要导入相应的依赖坐标。
  • @Entity:声明是一个实体。
  • @Table(name = “sys_user”):指明数据库中表的名字实现实体与数据表映射。
  • @Id:声明主键。
  • @Column(name = “user_id”):指明是表中的哪个字段。
  • @GeneratedValue(strategy = GenerationType.IDENTITY): GenerationType.IDENTITY:实现自增

启动项目

当我们启动项目的时候,数据库会自动创建表结构。
在这里插入图片描述
看一下数据库,发现表已经自动创建
在这里插入图片描述

对表操作

增加数据

首先需要创建UserRepository接口,要通过继承JpaRepository<T, ID>,提供实体类和id的类型。

/*** @author: lyd* @description: UserRepository* @Date: 2022/10/2*/
@Repository
public interface UserRepository extends JpaRepository<User, Long>, JpaSpecificationExecutor<User> {}

编写测试类
通过调用save方法即可完成保存操作

package com.jpa.demo.test;
import com.jpa.demo.domain.User;
import com.jpa.demo.repository.UserRepository;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.junit.jupiter.api.Test;
/*** @author: lyd* @description: 测试类* @Date: 2022/10/2*/
@SpringBootTest
public class JpaTest {@Autowiredprivate UserRepository userRepository;@Testvoid InsertTest() {User user = new User();user.setNickName("怒放吧德德");user.setUsername("lyd_code");user.setPassword("lyd_open");User save = userRepository.save(user);System.out.println(save);}
}

运行结果
在这里插入图片描述

查看数据

通过findAll就可以直接获取得到所有的数据。

@Test
void QueryTest() {List<User> all = userRepository.findAll();all.forEach(System.out::println);
}

就能在控制台中看到:
在这里插入图片描述

分页查找

jpa也是实现了分页操作的方法。通过PageRequest.of(0,1)可以完成分页。

@Test
void PageRequestTest() {userRepository.findAll(PageRequest.of(0,1)).forEach(System.out::println);
}

其底层对数据库的操作就会加上limit.
在这里插入图片描述

根据ID查找

返回是Optional类,User类型的数组

@Test
void FindByIdTest() {Optional<User> user = userRepository.findById(1L);System.out.println(user);
}

Java8 新特性Optional类
Optional 类新特性

Optional 类是一个可以为null的容器对象
如果值存在则isPresent()方法会返回true,调用get()方法会返回该对象
Optional 是个容器:它可以保存类型T的值,或仅保存null。Optional提供很多有用的方法,这样我们就不用显式进行空值检测
Optional 类的引入很好的解决空指针异常

方法命名规则自定义查询

方法命名规则查询就是根据方法的名字,就能创建查询。需要按照Spring Data JPA提供的方法命名规则定义方法的名称,就可以完成查询工作。Spring Data JPA在程序执行的时候会根据方法名称进行解析,并自动生成查询语句进行查询。通过jpa的规则要求来实现自己命名查询。
按照Spring Data JPA 定义的规则,查询方法以findBy开头,涉及条件查询时,条件的属性用条件关键字连接,要注意的是:条件属性首字母需大写。框架在进行方法名解析时,会先把方法名多余的前缀截取掉,然后对剩下部分进行解析。

编写Repository

根据一定的规则来编写方法命名。jpa会自动识别方法名称进行操作。

package com.jpa.demo.repository;import com.jpa.demo.domain.User;
import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
import org.springframework.stereotype.Repository;import java.util.List;/*** @author: lyd* @description: UserRepository* @Date: 2022/10/2*/
@Repository
public interface UserRepository extends JpaRepository<User, Long>, JpaSpecificationExecutor<User> {//根据nickname查询List<User> findAllByNickName(String str);
}

通过测试可以看到如下结果:
在这里插入图片描述
在编写repository的时候会有一些规则提示,也就是只有符合这些规则才能够被识别。这里只是展现了findxxx,实际上还有许多。
在这里插入图片描述
具体的关键字,使用方法和生产成SQL如下表所示
在这里插入图片描述

使用JPQL的方式查询(注解式SQL)

通过Query注解可以实现更多灵活的条件查询等SQL·操作。@Query 注解的使用非常简单,只需在方法上面标注该注解,同时提供一个JPQL查询语句即可,并且,在这些SQL语句中都是使用实体来进行操作,并非数据库的表结构来操作。

计数

通过Query可以写自己的SQL语句

@Query("select count(u) from User u") // 这里使用的是实体,并不是数据库的表名和字段
Integer countUser();

测试

@Test
void CountUser() {Integer integer = userRepository.countUser();System.out.println(integer);
}

结果
在这里插入图片描述

参数传递

在通过注解写SQL的时候,也需要是携带参数的,而Query通过?占位符来获得传来的参数数据,通过参数的位置定位哪个参数。

//修改数据 一定加上@Modifying 注解
@Transactional
@Modifying
@Query("update User set username=?1 where id=?2")
int updateUsernameByIds(String username,Long id);

结果
在这里插入图片描述
也可以使用原生的SQL,只需要加上 nativeQuery = true 即可。

实体关系对应

jpa中实体包括一对一,一对多,多对多,多对一关系,并且提供了相应的注解。接下来通过案例来实现关系。

一对一

首先创建部门表,使用户指向一个部门。

package com.jpa.demo.domain;
import lombok.Data;
import javax.persistence.*;
import java.io.Serializable;
/*** @author: lyd* @description: 部门* @Date: 2022/10/2*/
@Entity
@Data
@Table(name="sys_dept")
public class Dept implements Serializable {@Id@Column(name = "dept_id")@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String name;
}

在User实体中加字段,通过OneToOne注解完成。

@OneToOne(fetch = FetchType.LAZY, cascade = CascadeType.ALL) //设置关联操作为ALL
@JoinColumn(name = "dept_id")
private Dept dept;

这里的关联级别也是有多个,一般设置为all就行

  • ALL:所有操作都进行关联操作
  • PERSIST:插入操作时才进行关联操作
  • REMOVE:删除操作时才进行关联操作
  • MERGE:修改操作时才进行关联操作
    这样就可以编写测试
@Test
void OneToOneTest() {User user = new User();user.setNickName("爱国德德");user.setUsername("lyd_code_studio");user.setPassword("lyd_open");Dept dept = new Dept();dept.setName("研发部");user.setDept(dept);User save = userRepository.save(user);System.out.println(save);
}

通过测试可得数据库结果,
在这里插入图片描述
在这里插入图片描述
通过关联,会在两个表中进行插入,删除也是会删除两个表中数据。

一对多

@OneToMany:作用:建立一对多的关系映射属性:targetEntityClass:指定多的多方的类的字节码mappedBy:指定从表实体类中引用主表对象的名称。cascade:指定要使用的级联操作fetch:指定是否采用延迟加载orphanRemoval:是否使用孤儿删除

创建一个文章表来实现一对多关系,一个用户可以发表多个文章,一个文章只属于一个用户,因此在用户表中是一对多,在文章表就是多对一。
通过@ManyToOne注解完成关系

package com.jpa.demo.domain;
import lombok.Data;
import javax.persistence.*;
/*** @author: lyd* @description: 文章表* @Date: 2022/10/2*/
@Data
@Entity
@Table(name="yun_article")
public class Article {@Id@Column(name = "article_id")@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;@Column(nullable = false, length = 50) // 映射为字段,值不能为空private String title;@Basic(fetch = FetchType.LAZY) // 懒加载@Column(nullable = false) // 映射为字段,值不能为空private String content;@ManyToOne(cascade={CascadeType.MERGE,CascadeType.REFRESH},optional=false)//可选属性optional=false,表示user不能为空。删除文章,不影响用户@JoinColumn(name="user_id")//设置在article表中的关联字段(外键)private User user;
}

在user实体这边就是一对多关系,并且通过注解完成。

@OneToMany(mappedBy = "user",cascade=CascadeType.ALL,fetch=FetchType.LAZY)
//级联保存、更新、删除、刷新;延迟加载。当删除用户,会级联删除该用户的所有文章
//拥有mappedBy注解的实体类为关系被维护端
private Set<Article> articles;

测试,需要创建ArticleRepository

@Test
void OneToMany() {User user = userRepository.getById(1L);Article article1=new Article();article1.setTitle("如何写好博客");article1.setContent("如何写好博客");article1.setUser(user);articleRepository.save(article1);Article article2=new Article();article2.setTitle("如何更进一步学习");article2.setContent("不断地卷");article2.setUser(user);articleRepository.save(article2);
}

运行结果
在这里插入图片描述
数据库中
在这里插入图片描述

多对多

JPA中使用@ManyToMany来注解多对多的关系,由一个关联表来维护。这个关联表的表名默认是:主表名+下划线+从表名。(主表是指关系维护端对应的表,从表指关系被维护端对应的表)。这个关联表只有两个外键字段,分别指向主表ID和从表ID。字段的名称默认为:主表名+下划线+主表中的主键列名,从表名+下划线+从表中的主键列名。通过@JoinTable来实现中间表。
创建角色表

package com.jpa.demo.domain;import lombok.Data;import javax.persistence.*;
import java.io.Serializable;
import java.util.Set;/*** @author: lyd* @description: 角色表* @Date: 2022/10/2*/
@Data
@Entity
@Table(name = "sys_role")
public class Role implements Serializable {@Id@Column(name = "role_id")@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;@ManyToMany(mappedBy = "roles")private Set<User> users;private String name;@Column(name = "level")private Integer level = 3;private String description;
}

在用户表中

@ManyToMany
@JoinTable(name = "sys_users_roles",joinColumns = {@JoinColumn(name = "user_id",referencedColumnName = "user_id")},inverseJoinColumns = {@JoinColumn(name = "role_id",referencedColumnName = "role_id")})
private Set<Role> roles;

测试

@Test
void ManyToMany() {Role role = new Role();role.setName("USER");role.setLevel(2);role.setDescription("管理员");roleRepository.save(role);Set<Role> roles = new HashSet<>();roles.add(role);User user = new User();user.setPassword("lyd_code");user.setNickName("怒放德德");user.setUsername("lyd_code");user.setRoles(roles);User save = userRepository.save(user);System.out.println(save);}

在数据库中的中间表就能够看到如下数据。
在这里插入图片描述

文章推荐

  • 元动力文档合集 - SPRING-DATA-JPA
  • CSDN:https://blog.csdn.net/cmx1060220219/article/details/106259423
  • CSDN:https://blog.csdn.net/localhost01/article/details/83422893

👍创作不易,如有错误请指正,感谢观看!记得点赞哦!👍

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.luyixian.cn/news_show_17850.aspx

如若内容造成侵权/违法违规/事实不符,请联系dt猫网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

图解mysql(六)——内存篇

6 揭开Buffer Pool的面纱 6.1 为什么要有Buffer Pool&#xff1f; 虽然说 MySQL 的数据是存储在磁盘里的&#xff0c;但是也不能每次都从磁盘里面读取数据&#xff0c;这样性能是极差的。 要想提升查询性能&#xff0c;加个缓存就行了嘛。所以&#xff0c;当数据从磁盘中取出…

基于BootStrap的农业信息数据收集与管理平台设计

目 录 1 前言 1 1.1 研究背景 1 1.2 研究内容 1 1.3 研究意义 1 1.4 本论文工作和章节内容 2 2 系统环境与技术支持 3 2.1 系统环境 3 2.2 B/S架构 3 2.3 服务端技术 3 2.3.1 MVC模式 3 2.3.2 Spring框架 4 2.3.3 Hibernate框架 4 2.3.4 MySQL数据库 5 2.4 前端技术 5 2.4.1 Bo…

基于JavaWeb的企业出差费用报销管理系统设计与实现

目录 第一章 绪论 1 1.1 出差报销管理系统的开发背景 1 1.2设计目的与意义 1 第二章 系统需求分析 2 2.1 可行性分析 2 2.1.1 操作可行性 2 2.1.2 经济可行性 2 2.1.3 技术可行性 2 2.2方案的设计与比较 2 2.2.1 C/S设计结构和B/S设计结构比较 2 2.2.2 系统模式的设计 3 2.2.3系…

大二数据库实验-MySQL语句(Employee、Department、Salary)

实验所用到的的几张表&#xff1a; 显示Employee表中姓王的记录。 显示salary 表中InCome大于2000的数据。 显示salary 表中InCome在2500到3000之间的数据。 显示Employee表中在1968年下半年出生的数据。 分别显示三个表中总记录条数。 显示salary表中收入和支出总…

(附源码)计算机毕业设计ssm财务管理系统

毕设帮助&#xff0c;指导&#xff0c;本源码分享&#xff0c;调试部署(见文末) 3.2.1系统开发流程 财务管理系统开发时&#xff0c;首先进行需求分析&#xff0c;进而对系统进行总体的设计规划&#xff0c;设计系统功能模块&#xff0c;数据库的选择等&#xff0c;本系统的…

权限控制WAF绕过之木马混淆免杀

目录 前言&#xff1a; 绕过思路&#xff1a; &#xff08;一&#xff09;变量覆盖 &#xff08;二&#xff09;加密混杂 未加密前&#xff1a; 接口加密 加密后&#xff1a; WAF验证&#xff1a; &#xff08;三&#xff09;异或生成 0x01 原理&#xff1a; 0x02 适用…

线段树什么的最讨厌了

发现如果正着从一颗线段树搜到这一个区间,很难搜。所以考虑从一个区间搜出一颗线段树。 对于一个区间 \([l,r]\),他的父亲区间只可能是 \([2*l-r-2,r],[2*l-r-1,r],[l,2*r-l],[l,2*r-l-1]\) 四种情况。 发现无论往哪一种方向走,\(\frac{l}{r-l+1}\) 的值都会除以2.那么这么做…

建立对单片机/嵌入式启动、运行的整体认知

文章目录一、51单片机的启动过程二、STM32的完整启动流程分析1. 根据boot引脚决定三种启动模式2. 启动后bootloader做了什么&#xff1f;3. bootloader中对内存的搬移和初始化4. ISP、IAP、ICP三种烧录方式5. 参考资料从上电到启动&#xff0c;一文读懂STM32启动全流程1、直接上…

m基于FPGA的cordic算法实现,输出sin和cos波形(包括仿真录像)

目录 1.源码获取方式 2.算法描述 3.部分程序 4.部分仿真图预览 1.源码获取方式 使用版本matlab2022a 获取方式1&#xff1a; 点击下载链接&#xff08;解压密码C123456&#xff09;&#xff1a; m基于FPGA的cordic算法实现,输出sin和cos波形 获取方式2&#xff1a; 如…

程序员的数学课21 神经网络与深度学习:计算机是如何理解图像、文本和语音的?

在上一讲的最后&#xff0c;我们提到过“浅层模型”和“深层模型”。其实&#xff0c;人工智能的早期并没有“浅层模型”的概念&#xff0c;浅层模型是深度学习出现之后&#xff0c;与之对应而形成的概念。在浅层模型向深层模型转变的过程中&#xff0c;神经网络算法无疑是个催…

Vue2 生命周期

Vue 生命周期 概述在使用 Vue 时,我们需要执行一些 JS 代码。比如我们需要在页面中添加一个定时器来固定间隔更新时间。这时我们可能会想到直接在,Vue 实例外书写 JS 代码。这种方法能完成操作,但是 Vue 并不建议这样写。Vue 建议尽量在 Vue 实例中完成所有的操作。这时我们…

Hadoop3.X安装教程(Ubuntu)

前提:一台纯净的Ubuntu机器(虚拟机安装教程略) ctrl + alt + T 打开bash,全程使用bash指令进行,以hadoop 和 java 8为例 首先换源进入root账户 sudo su -升级软件列表 apt-get update安装vim apt install vim中途询问直接输入Y确认下载hadoop和java 创建/data mkdir /data…

半导体中的缺陷和位错能级

点缺陷&#xff1a; 在一定的温度下&#xff0c;组成晶体的格点原子在平衡位置附近做振动&#xff0c;这些振动就会有强有弱&#xff0c;这样会使得一部分原子可以获得足够的能量&#xff0c;而挣脱周围电子对它的束缚&#xff0c;挤入间隙位置&#xff0c;这样的结果就形成了…

211西北大学,计算机、软件学硕和专硕专业课都变难了!

西北大学位于陕西省西安市&#xff0c;是一所211大学。西北大学计算机学科评估B-&#xff0c;软件工程学科评估B&#xff0c;计算机实力在211大学中处于中上游水平&#xff0c;还算不错。西北大学前段时间公布了23考研的招生目录&#xff0c;我们来看一下&#xff1a;西北大学2…

Unity的UI框架

UI框架 UI框架的含义 含义&#xff1a;UI框架用于管理场景中所有的面板&#xff0c;负责控制面板之间的跳转 UI框架的意义 1、随着游戏系统的复杂化&#xff0c;UI控件越来越多&#xff0c;各个UI之间的直接通讯&#xff0c;已经UI与GameObject之间的关系会越来越复杂 2、代…

盘点一个Python自动化办公的实战案例

点击上方“Python共享之家”&#xff0c;进行关注回复“资源”即可获赠Python学习资料今日鸡汤岭猿同旦暮&#xff0c;江柳共风烟。大家好&#xff0c;我是皮皮。 一、前言前几天在Python钻石交流群【Hxy任我肥】问了一个Python自动化办公的问题&#xff0c;提问截图如下&am…

基于Vue+SSM+SpringCloudAlibaba的英雄管理系统

需求 前端技术&#xff1a;element-ui、vue后端技术&#xff1a;spring boot、spring cloud、mybatis plus、jwt项目要求&#xff1a; 前端&#xff1a;exam-war-fore-1217后端&#xff1a;exam-war-parent-1217端口要求&#xff1a; 注册中心&#xff1a;10086、10087 &#x…

福特、微软、槟榔-《软件方法》自测题解析019

DDD领域驱动设计批评文集>> 《软件方法》强化自测题集>> 《软件方法》各章合集>> 《软件方法》第2章自测题2 1 [单选题] 1999年11月的《财富》杂志题为“20世纪企业家”的文章&#xff0c;评选出了最能代表20世纪企业家精神的企业家─福特汽车的Henry F…

云原生|kubernetes|ingress-nginx插件部署以及简单的应用

前言&#xff1a; ingress直译&#xff1a;进口&#xff1b;入口&#xff1b;初切&#xff1b;进入&#xff1b;进入资格&#xff1b;进入权。在kubernetes中&#xff0c;它指的是网络入口。 ingress概述&#xff1a; 通俗来讲&#xff0c;Ingress和之前提到的Service、Depl…

Redis面试汇总笔记

在两个月前的学习中&#xff0c;我看过一个redis相关的讲解视频&#xff0c;是一个叫诸葛的老师&#xff0c;其中分为几层进行讲述&#xff0c;分别是数据类型、分布式锁、redis常见问题等。当时有记录一些内容&#xff0c;下面将按照顺序进行分享。 &#xff08;一&#xff0…