Spring Data JPA

news/2025/1/15 16:37:56/

文章目录

  • 一、Spring Data基础概念
  • 二、JPA与JDBC的相同与不同之处
  • 三、Hibernate & JPA快速搭建
    • 1.添加依赖
    • 2.实体类
    • 3.hibernate的配置文件 ——hibernate.cfg.xml
  • 四、测试——基于hibernate的持久化(单独使用)
  • 五、测试——基于JPA的持久化(单独使用)
    • 1.建立 /resources/META-INF/persistence.xml 配置文件
    • 2.案例
  • 六、JPA对象的四种状态
    • 临时状态
    • 持久状态
    • 删除状态
    • 游离状态
    • 方法下对象的不同状态情况
      • 1、persist(Object entity)
      • 2、merge(Object entity)
      • 3、refresh(Object entity)
      • 4、remove(Object entity)
  • 七、Spring Data JPA搭建 -- xml方式
    • 1.添加依赖
    • 2. spring.xml
    • 3. CRUD测试 -- spring
  • 八、Spring Data JPA搭建 -- JavaConfig 方式
    • 1.JavaConfig
  • 九、自定义操作
    • 1.JPQL和SQL
    • 2.规定方法名
    • 3.Query By Example 动态查询
    • 4.Specifications
    • 5.Query DSL

一、Spring Data基础概念

Spring Data:统一和简化对不同类型持久性存储(关系数据库系统和NoSQL数据存储)的访问。
特性:模板制作、对象/数据存储映射、Repository支持

二、JPA与JDBC的相同与不同之处

相同:
  1.都跟数据库操作有关,JPA是JDBC的升华。
  2.JPA和JDBC都是一组规范接口
不同之处:
  1.JDBC是由各个关系型数据库实现的,JPA是由ORM框架实现。
  2.JDBC使用SQL语句和数据库通信,JPA用面向对象方式,通过ORM框架来生成SQL,进行操作。
  3.JPA在JDBC之上,JPA也要依赖JDBC才能操作数据库。

三、Hibernate & JPA快速搭建

1.添加依赖

在这里插入图片描述

2.实体类

/*** 客户的实体类*      配置映射关系*   1.实体类和表的映射关系*      @Entity:声明实体类*      @Table:配置实体类和表的映射关系*          name : 配置数据库表的名称*   2.实体类中属性和表中字段的映射关系*/
@Data
@Entity
@Table(name = "tb_customer")
public class Customer {/*** @Id:声明主键的配置* @GeneratedValue:配置主键的生成策略*      strategy:*          GenerationType.IDENTITY :自增,mysql*                  底层数据库必须支持自动增长(底层数据库支持的自动增长方式,对id自增)*          GenerationType.SEQUENCE : 序列,oracle*                  底层数据库必须支持序列*          GenerationType.TABLE : jpa提供的一种机制,通过一张数据库表的形式帮助我们完成主键自增*          GenerationType.AUTO : 由程序自动的帮助我们选择主键生成策略* @Column:配置属性和字段的映射关系*      name:数据库表中字段的名称*/@Id@GeneratedValue(strategy = GenerationType.IDENTITY)@Column(name = "customer_id")private Long Id; //客户的主键@Column(name = "customer_name")private String name;//客户名称@Column(name="customer_age")private int age;//客户年龄@Column(name="customer_sex")private boolean sex;//客户性别@Column(name="customer_phone")private String phone;//客户的联系方式@Column(name="customer_address")private String address;//客户地址
}

@Entity 作为hiberbate实体类
@Tabel(name=“表名”) 映射的表名
@Id 声明主键
@GenerateValue 主键的生成策略
@Column(name=“字段名”) 表单的列名

3.hibernate的配置文件 ——hibernate.cfg.xml

<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC"-//Hibernate/Hibernate Configuration DTD//EN""http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration><session-factory><!--使用 Hibernate 自带的连接池配置--><property name="connection.driver_class">com.mysql.jdbc.Driver</property><property name="connection.url">jdbc:mysql://localhost:3306/bianchengbang_jdbc?characterEncoding=UTF-8</property><property name="hibernate.connection.username">root</property><property name="hibernate.connection.password">root</property><!--hibernate 方言--><property name="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</property><!--打印sql语句--><property name="hibernate.show_sql">true</property><!--格式化sql--><property name="hibernate.format_sql">true</property><!--表生成策略none        :不会创建表create      : 程序运行时创建数据库表(如果有表,先删除表再创建)update      :程序运行时创建表(如果有表,不会创建表)create-drop : 每次加载hibernate时根据model类生成表,但是sessionFactory一关闭,表就自动删除。validate    : 每次加载hibernate时,验证创建数据库表结构,只会和数据库中的表进行比较,不会创建新表,但是会插入新值--><property name="hbm2ddl.auto">update</property><!-- 加载映射文件 择一--><mapping resource="net/biancheng/www/mapping/User.hbm.xml"/><!-- 指定哪些pojo需要进行ORM映射 择一--><mapping class="com.test.pojo.Customer"/></session-factory>
</hibernate-configuration>

四、测试——基于hibernate的持久化(单独使用)

案例
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
懒查询
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

五、测试——基于JPA的持久化(单独使用)

1.建立 /resources/META-INF/persistence.xml 配置文件

依赖不用改,多加一个配置文件。IDEA有persistence模板:setting=》file and code Template=》JPA==》Deployment descriptors=》persistenceXX.xml

<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://java.sun.com/xml/ns/persistence" version="2.0"><!-- name:持久化单元名称,transaction-type:持久化单元事务类型(JTA:分布式事务管理,RESOURCE_LOCAL:本地事务管理) --><persistence-unit name="myJpa" transaction-type="RESOURCE_LOCAL"><!--jpa的实现方式,配置JPA服务提供商 --><provider>org.hibernate.jpa.HibernatePersistenceProvider</provider><!-- 可配可不配,如果配置了顺序不能错,必须在provider之后--><!--<class>com.test.pojo.Customer</class>--><!--可选配置:配置jpa实现方的配置信息--><properties><!-- 数据库信息配置:数据库驱动、数据库地址、数据库账户、数据库密码 --><property name="hibernate.connection.driver_class" value="com.mysql.jdbc.Driver"/><property name="hibernate.connection.url" value="jdbc:mysql://127.0.0.1:3306/hibernate_jpa?characterEncoding=UTF-8"/><property name="hibernate.connection.username" value="root"/><property name="hibernate.connection.password" value="password"/><!-- 配置JPA服务提供商可选参数 --><property name="hibernate.show_sql" value="true" /><!-- 自动显示sql --><property name="hibernate.format_sql" value="true"/><!-- 格式化sql --><!-- 自动创建数据库表:none        :不会创建表create      : 程序运行时创建数据库表(如果有表,先删除表再创建)update      :程序运行时创建表(如果有表,不会创建表)create-drop : 每次加载hibernate时根据model类生成表,但是sessionFactory一关闭,表就自动删除。validate    : 每次加载hibernate时,验证创建数据库表结构,只会和数据库中的表进行比较,不会创建新表,但是会插入新值。--><property name="hibernate.hbm2ddl.auto" value="update" /></properties></persistence-unit>
</persistence>

2.案例

 /**
* 测试jpa的保存    案例:保存一个客户到数据库中
* Jpa的操作步骤
* 1.加载配置文件创建工厂(实体管理器工厂)对象
* 2.通过实体管理器工厂获取实体管理器
* 3.获取事务对象,开启事务
* 4.完成增删改查操作
* 5.提交事务(回滚事务)
* 6.释放资源
*/
@Test
public void testSave() {//1.加载配置文件创建工厂(实体管理器工厂)对象EntityManagerFactory factory = Persistence.createEntityManagerFactory("myJpa");//2.通过实体管理器工厂获取实体管理器EntityManager em = factory.createEntityManager();//3.获取事务对象,开启事务EntityTransaction tx = em.getTransaction();tx.begin();//4.完成增删改查操作:保存一个客户到数据库中Customer customer = new Customer();customer.setName("Sam");customer.setAddress("Beijing");//保存操作em.persist(customer);//5.提交事务tx.commit();//6.释放资源em.close();factory.close();
}---------------------------------api
//立即查询
Customer customer = em.find(Custormer.class,1L);
//懒查询
Customer customer = em.getReference(Custormer.class,1L);
//更新 指定id会先查询数据是否有变化,有变化则更新,不指定id会插入一条数据
Customer customer = new Customer();
customer.setCustId(5L);
customer.setCustName("dfg");
em.merge(customer);
//更新 -- JPQL 不会进行查询 使用实体类名和属性名
String jpql = "Update Customer set name=:name where id=:id ";
em.createQuery(jpql).setParameter("name","李四").setParameter("id","5L").executeUpdate();
//更新 -- 使用SQL 使用表名和字段名
String sql = "Update tb_customer set name=:name where id=:id ";
em.createNativeQuery(sql).setParameter("name","李四11").setParameter("id","5L").executeUpdate();
//删除 先查再删除
//不能删除游离数据,只能删除持久化数据(从数据库里查出来的数据)
Customer customer =  em.find(Custormer.class,5L);
em.remove(customer);

六、JPA对象的四种状态

临时状态

刚创建出来的对象,它此时还没与entityManager发生关系,没有被持久化,不处于entityManager中的对象。

持久状态

与entityManager发生关系,已经被持久化,是数据库实在的记录,比如执行了persist()、find()、merge()。
情况:
持久状态进行了修改会同步数据库;把数据库里的记录查询出来做修改并没有执行保存操作,也会把更新数据的操作同步到数据库。

删除状态

执行remove()方法,事物提交之前。

游离状态

游离状态是提交到数据库后,事务commit后实体的状态。比如该对象commit之后在数据库有对应的数据,但是它还没与entityManager发生关系。此时实体的属性可以做任何改变,也不会同步到数据库,因为事务已经提交了。
在这里插入图片描述

方法下对象的不同状态情况

1、persist(Object entity)

persist方法可以将实例转换为managed(托管)状态,在调用flush()方法或提交事物之后 ,实例将会被插入到数据库中。
对不同状态下的实例A,persist会产生以下操作:
1)如果A是一个new状态的实体,它将会转为managed状态;
2)如果A是一个managed状态的实体,它的状态不会发生任何改变,但是系统仍会在数据库执行 insert 操作;
3)如果A是一个removed(删除)状态的实体,它将会转换为受控状态;
4)如果A是一个detached(分离)状态的实体,该方法会抛出IllegalArgumentException异常,具体异常根据不同的JPA实现有关。

2、merge(Object entity)

merge方法的主要作用是将用户对一个detached状态实体的修改进行归档,归档后将产生一个新的managed状态对象。
对不同状态下的实例A,merge会产生以下操作:
1)如果A是一个detached状态的实体,该方法会将A的修改提交到数据库,并返回一个新的managed状态的实例A2;
2)如果A是一个new状态的实体,该方法会产生一个根据A产生的managed状态实体A2;
3)如果A是一个managed状态的实体,它的状态不会发生任何改变,但是系统仍会在数据库执行update操作。
4)如果A是一个removed状态的实体,该方法会抛出IllegalArgumentException异常。

3、refresh(Object entity)

refresh方法可以保证当前的实例与数据库中的实例内容一致。
对不同状态下的实例A,refresh会产生以下操作:
1)如果A是一个new状态的实体,A的状态不会发生任何改变,但系统仍会在数据库中执行delete语句;
2)如果A是一个managed状态的实体,它的属性将会和数据库中的数据同步;
3)如果A是一个removed状态的实体,该方法将会抛出异常:Entity not managed
4)如果A是一个detached状态的实体,该方法将会抛出异常;

4、remove(Object entity)

remove方法可以将实体转换为removed状态,并且在调用flush()方法或提交事物后删除数据库中的数据。
对不同状态下的实例A,remove会产生以下操作:
1)如果A是一个new状态的实体,A的状态不会发生任何改变,但系统仍会在数据库中执行delete语句;
2)如果A是一个managed状态的实体,它的状态会转换为removed;
3)如果A是一个removed状态的实体,不会发生任何操作;
4)如果A是一个detached状态的实体,该方法将会抛出异常;

七、Spring Data JPA搭建 – xml方式

1.添加依赖

父项目
添加spring data的统一版本管理,它可以统一子项目的Spring Data JDBC、Spring Data JPA 、Spring Data Redis等版本。

<dependencyManagement><dependencies><dependency><groupId>org.springframework.data</groupId><artifactId>spring-data-bom</artifactId><version>2022.0.2</version><scope>import</scope><type>pom</type></dependency></dependencies>
</dependencyManagement>

子项目
spring 项目需要另外添加hibernate依赖,spring boot项目不需要。

<dependencies><dependency><groupId>org.springframework.data</groupId><artifactId>spring-data-jpa</artifactId></dependency>
<dependencies>

2. spring.xml

整合jpa
在这里插入图片描述
在这里插入图片描述
< jpa:repositories />对应@EnableJpaRepositories注解
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

3. CRUD测试 – spring

定义接口继承CrudRepository接口

public interface CrudRepository<T, ID> extends Repository<T, ID> {}

在这里插入图片描述
查询
在这里插入图片描述
插入
在这里插入图片描述
更新 先查再更新
在这里插入图片描述
删除 先查再删除,把new出来的实例的游离状态改为持久状态
在这里插入图片描述

八、Spring Data JPA搭建 – JavaConfig 方式

1.JavaConfig

//标记当前类为配置类
@Configuration       
//启动jpa  相当于xml的<jpa:repository/>标签   basePackage:指定数据返回层.接口
@EnableJpaRepositories(basePackage="com.test.repositories")
//开启事务
@EnableTransactionManagement
class ApplicationConfig {@Beanpublic DataSource dataSource() {DruidDataSource dataSource = new DruidDataSource();dataSource.setUserName("root");dataSource.setPassWord("123456");dataSource.setDriverClassName("com.mysql.jdbc.Driver");dataSource.setUrl("jdbc:mysql://127.0.0.1:3306/hibernate_jpa?characterEncoding=UTF-8");return dataSource;//使用内嵌内存数据库/**EmbeddedDatabaseBuilder builder = new EmbeddedDatabaseBuilder();return builder.setType(EmbeddedDatabaseType.HSQL).build();**/}@Beanpublic LocalContainerEntityManagerFactoryBean entityManagerFactory() {HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();vendorAdapter.setGenerateDdl(true);vendorAdapter.setShowSql(true);LocalContainerEntityManagerFactoryBean factory = new LocalContainerEntityManagerFactoryBean();factory.setJpaVendorAdapter(vendorAdapter);//指定实体类的包factory.setPackagesToScan("com.acme.domain");factory.setDataSource(dataSource());return factory;}@Beanpublic PlatformTransactionManager transactionManager(EntityManagerFactory entityManagerFactory) {JpaTransactionManager txManager = new JpaTransactionManager();txManager.setEntityManagerFactory(entityManagerFactory);return txManager;}
}

测试使用
在这里插入图片描述

九、自定义操作

1.JPQL和SQL

JPQL定义,参数可以使用索引指定,或参数名结合@Param注解指定。
在这里插入图片描述
在这里插入图片描述
用 法
在这里插入图片描述
增删改需要开启事务,@Transactional通常放在业务逻辑层上声明。
在这里插入图片描述
JPQL是不支持新增的,用Hibernate实现伪插入,只能插入从别的地方查到的值。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

SQL
在这里插入图片描述

2.规定方法名

根据提供的主题关键字(前缀)决定方法作用,谓词关键字和修饰符决定查询条件。只支持查询和删除。官网使用说明,有详细使用方法的
例:
findBy(关键字)+ 属性名称(首字母大写)+ 查询条件(首字母大写)
在这里插入图片描述

3.Query By Example 动态查询

只支持查询,只支持字符串(开头、包含、结尾、正则匹配),不支持嵌套或分组的属性约束。官网使用说明
实现:

//继承QueryByExampleExecutor接口
public interface CustomerRepository extends PagingAndSortingRepository<Customer, Long> ,QueryByExampleExecutor<Customer>{}

测试:
通过customer对象来进行筛选。
在这里插入图片描述
通过匹配器进行条件的限制。
在这里插入图片描述

4.Specifications

对所有类型支持。
步骤:
1、通过root拿到需要设置条件的字段
2、通过CriteraBuilder设置不同类型条件
3、组合条件

实现:

//继承JpaSpecificationExecutor接口
public interface CustomerRepository extends CrudRepository<Customer, Long>, JpaSpecificationExecutor<Customer> {} 

测试:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

5.Query DSL

通用查询框架,借助QueryDSL可以在任何支持的ORM框架或SQL平台上以通用API方式构建查询。官网使用说明
实现:
在这里插入图片描述

//继承QuerydslPredicateExecutor
interface UserRepository extends CrudRepository<User, Long>, QuerydslPredicateExecutor<User> {
}//官方例子,能直接实体类user.firstname是添加了插件
Predicate predicate = user.firstname.equalsIgnoreCase("dave").and(user.lastname.startsWithIgnoreCase("mathews"));userRepository.findAll(predicate);

使用:
在这里插入图片描述
在这里插入图片描述


http://www.ppmy.cn/news/31837.html

相关文章

RHCSA-用户和组管理和文件系统权限(3.11)

目录 用户&#xff08;UID&#xff09; 用户类别&#xff08;UID&#xff09;&#xff1a; 用户的增删改查&#xff1a; 修改用户密码&#xff1a; 查看用户是否存在&#xff1a; 组&#xff08;GID&#xff09; 组的增删改查&#xff1a; 设置组密码&#xff1a; 用户…

C语言--字符串函数1

目录前言strlenstrlen的模拟实现strcpystrcatstrcat的模拟实现strcmpstrcmp的模拟实现strncpystrncatstrncmpstrstrstrchr和strrchrstrstr的模拟实现前言 本章我们将重点介绍处理字符和字符串的库函数的使用和注意事项。 strlen 我们先来看一个我们最熟悉的求字符串长度的库…

搭建一个中心化的定时服务

1. 背景 在物联网络&#xff0c;很多设备之间都在进行交互&#xff0c;其中云端在远程交流中起到了很重要的作用。比如&#xff0c;一台设备想进行调温&#xff0c;但是需要知道此时房间的温度&#xff0c;那就需要定时去查询传感器测出来的房间温度&#xff0c;如果温度过高&a…

2023年江苏省职业院校技能大赛中职网络安全赛项试卷-教师组任务书

2023年江苏省职业院校技能大赛中职网络安全赛项试卷-教师组任务书 一、竞赛时间 9:00-12:00&#xff0c;12:00-15:00&#xff0c;15:00-17:00共计8小时。 二、竞赛阶段 竞赛阶段 任务阶段 竞赛任务 竞赛时间 分值 第一阶段 基础设施设置与安全加固、网络安全事件响应、数…

互联网新理念,对于WEB 3.0 你怎么看?

WEB 3.0 这个名词走进大众视野已经有一段时间了&#xff0c;也曾在各个圈子里火热一时&#xff0c;至今各大互联网企业任旧在 WEB 3.0 上不断探索。但关于 WEB 3.0 是什么这个问题&#xff0c;其实大部分人都没有一个比较明确的认知&#xff0c;包括区块链和元宇宙等相关行业的…

【数据结构初阶】单链表面试题|内含链表带环问题

目录 前言 链表面试题 1. 删除链表中等于给定值 val 的所有节点。oj链接 2.反转一个单链表。oj链接 3. 给定一个带有头结点 head 的非空单链表&#xff0c;返回链表的中间结点。如果有两个中间结点&#xff0c;则返回第二个中间结点。oj链接 4. 输入一个链表&#xff0c;…

flink入门-流处理

入门需要掌握&#xff1a;从入门demo理解、flink 系统架构&#xff08;看几个关键组件&#xff09;、安装、使用flink的命令跑jar包flink的webUI 界面的监控、常见错误、调优 一、入门demo&#xff1a;统计单词个数 0、单词txt 文本内容(words.txt)&#xff1a; hello world …

【Linux】线程实例 | 简单线程池

今天来写一个简单版本的线程池 1.啥是线程池 池塘&#xff0c;顾名思义&#xff0c;线程池就是一个有很多线程的容器。 我们只需要把任务交到这个线程的池子里面&#xff0c;其就能帮我们多线程执行任务&#xff0c;计算出结果。 与阻塞队列不同的是&#xff0c;线程池中内有…