当前位置:   article > 正文

SpringData JPA 快速入门案例详解

springdata jpa

SpringData JPA

JPA 简介:

JPA(Java Persistence API)是 Java 持久层规范,定义了一些列 ORM 接口,它本身是不能直接使用的,因为接口需要实现才能使用,Hibernate 框架就是实现 JPA 规范的框架。

SpringData JPA 简介

Spring Data JPA 是 Spring 框架提供的对 JPA 规范的抽象,通过约定的命名规范完成持久层接口的编写,在不需要实现接口的情况下,就可以实现对数据库的操作。简单理解就是通过 Spring Data JPA,你只需要定义接口而不需要实现,就能完成 CRUD 操作。

Spring Data JPA 本身并不是一个具体实现,它只是一个抽象层,底层还是需要 Hibernate 这样的 JPA 实现来提供支持。

Spring Data JPA 是一个全自动化的 ORM 框架,底层是 Hibernate 框架提供支持,开发者只需要调用接口方法即可,不必关心 SQL 语句和结果集的解析,非常方便

1. 添加 pom.xml 依赖

  1. <parent>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-parent</artifactId>
  4. <version>2.3.2.RELEASE</version>
  5. <relativePath/>
  6. </parent>
  7. <dependencies>
  8. <dependency>
  9. <groupId>org.springframework.boot</groupId>
  10. <artifactId>spring-boot-starter-data-jpa</artifactId>
  11. </dependency>
  12. <dependency>
  13. <groupId>org.projectlombok</groupId>
  14. <artifactId>lombok</artifactId>
  15. <optional>true</optional>
  16. </dependency>
  17. <dependency>
  18. <groupId>mysql</groupId>
  19. <artifactId>mysql-connector-java</artifactId>
  20. <scope>runtime</scope>
  21. </dependency>
  22. <dependency>
  23. <groupId>org.springframework.boot</groupId>
  24. <artifactId>spring-boot-starter-test</artifactId>
  25. <scope>test</scope>
  26. </dependency>
  27. </dependencies>

2. 配置数据源

/src/main/resources/application.properties 配置文件中

  1. ## 配置数据源
  2. spring:
  3. datasource:
  4. url: jdbc:mysql://127.0.0.1:3306/wdzldb?useSSL=false&serverTimezone=Asia/Shanghai
  5. username: root
  6. driver-class-name: com.mysql.cj.jdbc.Driver
  7. password: root
  8. jpa:
  9. show-sql: true
  10. properties:
  11. hibernate:
  12. format_sql: true # 格式化SQL
  13. # 生成SQL 驼峰命名属性,默认加下划线的情况,下面配置可以去掉下划线。
  14. hibernate:
  15. naming:
  16. physical-strategy: org.hibernate.boot.model.naming.PhysicalNamingStrategyStandardImpl

3. 编写实体类

注意:这里使用的注解和 SpringData JDBC 的不同,属性名出现 驼峰命名的,也会默认映射为 book_name 的形式,此时需要通过 @Column 注解来映射。也可以在配置文件中统一配置关闭下划线。

注意:常见错误

  1. Caused by: java.lang.IllegalArgumentException: Not a managed type: class com.wdzl.pojo.Book
  2. at org.hibernate.metamodel.internal.MetamodelImpl.managedType(MetamodelImpl.java:582)
  3. at org.hibernate.metamodel.internal.MetamodelImpl.managedType(MetamodelImpl.java:85)

原因:实体类没有指定@Entry 注解。 添加注解后,会显示编译错误,要求必须指定主键 @Id

主键需要指定主键生成策略,否则默认的是

GenerationType strategy() default GenerationType.AUTO;

默认的策略是序列,应用于 Oracle 数据库

  1. package com.wdzl.pojo;
  2. import lombok.Data;
  3. import javax.persistence.*;
  4. import java.util.Date;
  5. @Data
  6. @Entity //要求必须有@Id
  7. public class Book {
  8. @Id
  9. @GeneratedValue(strategy = GenerationType.IDENTITY)
  10. @Column(name="bookid")
  11. private Integer bookId;
  12. @Column(name="bookname")
  13. private String bookName;
  14. private Float price;
  15. @Column
  16. private Date pubdate;
  17. private String author;
  18. }

4. 编写持久层接口

注意:这里继承的接口是 JpaRepository

在 JpaRepository 接口中包含了继承的 CRUD 方法,如果需要组合条件查询,可以遵循 “约定大于配置” 策略,实现组合条件查询。

注意异常:

  1. Caused by: java.lang.IllegalArgumentException: Not a managed type: class java.lang.Object
  2. at org.hibernate.metamodel.internal.MetamodelImpl.managedType(MetamodelImpl.java:582) ~[hibernate-core-5.4.32.Final.jar:5.4.32.Final]
  3. at org.hibernate.metamodel.internal.MetamodelImpl.managedType(MetamodelImpl.java:85) ~[hibernate-core-5.4.32.Final.jar:5.4.32.Final]

原因:没有指定泛型: extends JpaRepository<Book,Integer>

IBookDao 类代码如下:

下面方法通过默认约定名来实现

  1. import com.wdzl.pojo.Book;
  2. import org.springframework.data.jpa.repository.JpaRepository;
  3. import org.springframework.stereotype.Repository;
  4. import java.util.List;
  5. @Repository
  6. public interface IBookDao extends JpaRepository<Book,Integer> {
  7. // bookname=?
  8. List<Book> findBookByBookName(String bookName);
  9. // 等价于 between price ? and ?
  10. List<Book> findBookByPriceBetween(Float price1,Float price2);
  11. //等价于 price>?
  12. List<Book> findBookByPriceAfter(Float price);
  13. //bookName like ? ==='%实%'
  14. List<Book> findBookByBookNameContaining(String bookName);
  15. //bookName like ? ==='%实%'
  16. List<Book> findBookByBookNameContains(String bookName);
  17. // bookName like '%入门'
  18. List<Book> findBookByBookNameEndsWith(String bookName);
  19. }

也可以通过 HQL 来定义方法的查询,因为 SpringData Jpa 通过 Hibernate 实现的。

  1. //注意下面是 HQL
  2. @Query("from Book where bookName like :bookname and price > :price")
  3. List<Book> findCondtion(String bookname,Float price);

注意:不支持 select *

5. 编写业务类

  1. @Service("bookService")
  2. public class BookService {
  3. @Autowired
  4. private IBookDao bookDao;
  5. public List<Book> queryAll(){
  6. return bookDao.findAll(); //这里不同jdbc 这里直接返回 list
  7. }
  8. public void save(Book book){
  9. bookDao.save(book);
  10. }
  11. public long count(){
  12. return bookDao.count();
  13. }
  14. }

6. 编写测试类

注意:测试类 一定要和启动类同包,这里是代替了启动类

  1. package com.wdzl;
  2. import com.wdzl.pojo.Book;
  3. import com.wdzl.service.BookService;
  4. import org.junit.jupiter.api.Test;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.boot.autoconfigure.SpringBootApplication;
  7. import org.springframework.boot.test.context.SpringBootTest;
  8. import java.util.Date;
  9. @SpringBootTest
  10. @SpringBootApplication //注意这里一定要和被测试类在同包 即启动类的位置
  11. public class TestApplication {
  12. @Autowired
  13. private BookService bookService;
  14. @Test
  15. public void testQuery(){
  16. bookService.queryAll().forEach(book -> System.out.println(book));
  17. }
  18. @Test
  19. public void save(){
  20. Book book = new Book();
  21. book.setPubdate(new Date());
  22. book.setBookName("Java入门与提高");
  23. book.setPrice(34F);
  24. bookService.save(book); //主键不存在 执行插入
  25. }
  26. @Test
  27. public void update(){
  28. Book book = new Book();
  29. book.setBookId(9);
  30. book.setPubdate(new Date());
  31. book.setBookName("Java基础入门");
  32. book.setPrice(24F);
  33. bookService.save(book); //主键存在执行修改
  34. }
  35. }

注意:

在执行 save() 保存时,如果设置了主键,则会先进行查询,再根据查询结果来执行操作,如果查询记录存在,则修改。否则不存在则插入。

如果主键策略没有指定自动增长,则在执行 save() 时,必须要设置主键。否则执行失败。

下面是设置了主键,但是数据库表中不存在的执行效果:

先执行了查询,再执行了 insert 插入操作

  1. Hibernate:
  2. select
  3. book0_.bookid as bookid1_0_0_,
  4. book0_.author as author2_0_0_,
  5. book0_.bookname as bookname3_0_0_,
  6. book0_.price as price4_0_0_,
  7. book0_.pubdate as pubdate5_0_0_
  8. from
  9. book book0_
  10. where
  11. book0_.bookid=?
  12. Hibernate:
  13. insert
  14. into
  15. book
  16. (author, bookname, price, pubdate)
  17. values
  18. (?, ?, ?, ?)

注意:

从上面控制台打印的结果来看,首先底层使用的Hibernate实现,

其次,先根据指定主键执行了查询来判断是否存在记录。

最后因为没有查询到记录,而执行 insert 操作。

7. JpaRepository 中方法详解

1) 接口源码中方法
  1. @NoRepositoryBean
  2. public interface JpaRepository<T, ID> extends PagingAndSortingRepository<T, ID>, QueryByExampleExecutor<T> {
  3. List<T> findAll(); //查询全部
  4. List<T> findAll(Sort var1); //排序查询
  5. List<T> findAllById(Iterable<ID> var1); //按多个主键查询
  6. <S extends T> List<S> saveAll(Iterable<S> var1);
  7. void flush();
  8. <S extends T> S saveAndFlush(S var1);
  9. <S extends T> List<S> saveAllAndFlush(Iterable<S> var1);
  10. /** @deprecated */
  11. @Deprecated
  12. default void deleteInBatch(Iterable<T> entities) {
  13. this.deleteAllInBatch(entities);
  14. }
  15. void deleteAllInBatch(Iterable<T> var1);
  16. void deleteAllByIdInBatch(Iterable<ID> var1);
  17. void deleteAllInBatch();
  18. /** @deprecated */
  19. @Deprecated
  20. T getOne(ID var1);
  21. T getById(ID var1);
  22. <S extends T> List<S> findAll(Example<S> var1);
  23. <S extends T> List<S> findAll(Example<S> var1, Sort var2);
  24. }
2) 添加:
  1. public void save(){
  2. Book book = new Book();
  3. book.setBookName("MySQL性能优化");
  4. book.setPrice(45f);
  5. bookDao.save(book);
  6. }
3) 修改:
  1. public void update(){
  2. Book book = bookDao.getById(25);
  3. book.setPubdate(new Date());
  4. book.setBookName("MySQL性能优化");
  5. book.setPrice(45f);
  6. bookDao.save(book);
  7. }

注意:

修改使用save方法,如果新实例化的对象,如果有主键且表中存在,则直接修改操作

如果实例化对象,同时指定主键,但不存在,也会自动插入操作。

但是,如果先查询,再修改,则默认情况下会抛出异常:

org.hibernate.LazyInitializationException: could not initialize proxy [com.wdzl.pojo.Book#25] - no Session

原因是默认使用了延时加载策略,这样在查询后已经关闭了session,但有修改操作了没加载的属性。导致异常。

解决办法:

在实体类上添加注解,关闭延时加载 : @Proxy(lazy = false) 或 放入事务单元

  1. @Data
  2. @Entity
  3. @Proxy(lazy = false)
  4. public class Book {
  5. @Id
  6. @GeneratedValue(strategy = GenerationType.IDENTITY)
  7. private Integer bookId;
  8. private String bookName;
  9. private Float price;
  10. private Date pubdate;
  11. private String author;
  12. }

第二种修改方式:使用HQL语句修改,但不支持insert .

注意:事务和允许修改注解 @Transactional 和 @Modifying

  1. @Modifying
  2. @Query("update Book set price=:price where bookId=:bookid") //类名和属性名
  3. @Transactional
  4. int update(Float price,Integer bookid);
4) 删除

源码如下:

  1. @Deprecated
  2. default void deleteInBatch(Iterable<T> entities) { //已经过时,不用关注
  3. this.deleteAllInBatch(entities);
  4. }
  5. void deleteAllInBatch(Iterable<T> var1); //按对象批量删除
  6. void deleteAllByIdInBatch(Iterable<ID> var1); //按主键批量删除
  7. void deleteAllInBatch(); //全部删除

案例应用:

  1. //批量删除,
  2. // 生成SQL: Hibernate: delete from Book where bookId in (? , ? , ? , ?)
  3. public void delByBatch(){
  4. List<Integer> ids = Arrays.asList(3,23,12,45);
  5. bookDao.deleteAllByIdInBatch(ids);
  6. }

也可以使用 HQL语句来删除,也需要事务支持。

  1. @Modifying
  2. @Query("delete from Book where bookName=:bookName")
  3. @Transactional
  4. int delete(String bookName);

也支持按约定规则命名方式来删除,注意:也需要事务支持

  1. @Transactional
  2. void deleteBookByAuthor(String author);
  3. // between
  4. List<Book> findBooksByPriceBetween(Float min,Float max);
  5. // 模糊查询 like 语句
  6. List<Book> findBooksByBookNameLike(String bookName);

生成 SQL如下:

  1. Hibernate: select book0_.bookId as bookid1_0_, book0_.author as author2_0_, book0_.bookName as bookname3_0_, book0_.price as price4_0_, book0_.pubdate as pubdate5_0_ from Book book0_ where book0_.author=?
  2. Hibernate: delete from Book where bookId=?

从上面可以看出,在删除时,先做查询,再做删除。如果查询不到,则不执行删除操作;

5) 查询
  1. List<T> findAll(); //查询全部
  2. List<T> findAll(Sort var1); //排序查询
  3. List<T> findAllById(Iterable<ID> var1); //按多个主键查询
  4. T getById(ID var1);
  5. <S extends T> List<S> findAll(Example<S> var1); // 条件查询
  6. <S extends T> List<S> findAll(Example<S> var1, Sort var2);
条件查询:
  1. //条件查询
  2. public void findBookByExample(){
  3. Book book = new Book();
  4. book.setBookName("Java");
  5. book.setPrice(23f);
  6. book.setBookId(23);
  7. Example<Book> bookExample = Example.of(book);
  8. bookDao.findAll(bookExample);
  9. }
排序:
  1. //可以按指定属性排序
  2. public void querySort(){
  3. //Order.desc()|asc()分别指定降序和升序
  4. Sort price = Sort.by(Sort.Order.desc("price"));
  5. List<Book> all = bookDao.findAll(price);
  6. all.forEach(System.out::println);
  7. }
分页:
  1. public void queryByPage(){
  2. //分页查询 第一个参数:页号(从0开始),第二参数:每页显示记录数
  3. Pageable pageable = PageRequest.of(2,2);
  4. bookDao.findAll(pageable).forEach(System.out::println);
  5. }
6) 多对一 和 一对多

下面配置多对一关联关系。图书类关联依赖作者类。

注意:不使用@Data 为了避免 toString(),在双向多对一时,产生死循环。

图书类:

  1. @Getter
  2. @Setter
  3. @Entity
  4. //@Proxy(lazy = false)
  5. public class Book {
  6. @Id
  7. @GeneratedValue(strategy = GenerationType.IDENTITY)
  8. private Integer bookId;
  9. private String bookName;
  10. private Float price;
  11. private Date pubdate;
  12. // private String author;
  13. //
  14. @JoinColumn(name="author") //关联的外键
  15. @ManyToOne(fetch = FetchType.LAZY)
  16. private Author author;
  17. }

作者类:

  1. @Entity(name="usertbl")
  2. @Getter
  3. @Setter
  4. public class Author {
  5. @Id
  6. private Integer userid;
  7. private String username;
  8. private String email;
  9. @OneToMany(mappedBy = "author",fetch = FetchType.LAZY)
  10. private Set<Book> books = new HashSet<>();
  11. }

图书关联查询作者:

  1. @Transactional
  2. public void manyToOne(){
  3. Book book = bookDao.getById(22);
  4. System.out.println(book.getBookName()+"=="+book.getAuthor());
  5. }

作者关联查询图书:

  1. @Transactional
  2. public void query(){
  3. Author author = authorDao.findAuthorByUserid(5);
  4. Set<Book> books = author.getBooks();
  5. books.forEach(s-> System.out.println(s.getBookName()));
  6. }

常见错误:

org.hibernate.LazyInitializationException: could not initialize proxy [com.wdzl.pojo.Book#22] - no Session

原因:

默认关联对象是延时加载的,在查询对象后默认关闭session,在后面使用到关联属性或对象时,再去查询时,已经不存在session可用了。所以抛出异常。

解决办法:

可以使用事务,让整个方法在一个事务单元中,使用完毕后再自动关闭session.

7)HQL

HQL(Hibernate Query Language)是 Hibernate 框架提供的面向对象的查询语言,它是 SQL 的一个超集,主要用来与关系型数据库进行交互,但它使用的是面向对象的方式来表达查询。HQL 查询的对象不再是数据库表,而是实体类及其属性。

  1. /**
  2. * HQL
  3. */
  4. @Repository
  5. public interface IBookDao extends JpaRepository<Book,Integer> {
  6. //支持HQL
  7. // @Query("select book from Book book")
  8. @Query("from Book")
  9. List<Book> queryAllBook();
  10. @Query("select bookId,bookName,author from Book")
  11. List<Object[]> queryAll();
  12. /**
  13. * 如果要修改,需要添加注解 @Modifying @Transactional
  14. * @param price
  15. * @param bookid
  16. * @return
  17. */
  18. @Modifying
  19. @Query("update Book set price=:price where bookId=:bookid")
  20. @Transactional
  21. int update(Float price,Integer bookid);
  22. @Modifying
  23. @Query("delete from Book where bookName=:bookName")
  24. @Transactional
  25. int delete(String bookName);
  26. @Transactional
  27. void deleteBookByBookName(String bookname);
  28. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Gausst松鼠会/article/detail/526280
推荐阅读
相关标签
  

闽ICP备14008679号