草庐IT

SpringBoot JPA开发

ElliotG 2023-03-28 原文

1. 什么是JPA

JPA是Java Persistence API的简称,是针对POJO类的ORM Java规范。

  • Spring Data JPA
    Spring Data JPA和JPA及Hibernate之间是紧密关联的。
    JPA是Java ORM的规范API。
    Hibernate则实现了JPA的这种规范。
    Spring Data JPA是通用JPA规范的一种补充实现,它不仅提供了JPA的实现,同时还基于Spring的特性提供了额外的辅助功能。

 

2. Spring Data JPA的功能

  • Spring Data JPA并没有完整地实现JPA
    Spring Data JPA的底层实现可以使用不同的服务提供者,
    包括但不限于Hibernate, Eclipse Link和Open JPA等,这些服务提供者才是真正的JPA规范实现者。
    Spring Data JPA只是提供了一种上层封装,为开发者提供了顶层接口规范。

  • 支持Repositories模式(具体什么是Repository请参考领域驱动设计DDD相关的概念)

  • 提供了audit功能

  • 支持Querydsl断言(Predicate)

  • 分页,排序,动态查询语句执行

  • 支持@Query注解

  • 支持xml映射定义

Spring Data JPA提供了一种约定大于配置的repositories实现,开发者可以减少大部分简单而又重复的CRUD操作,在传统的开发方式(eg: mybatis)中,开发者需要针对每个entity实体编写不用的CRUD操作。
Spring Data JPA提供的repositories机制为所有的entity实体提供了通用的解决方案,开发者只需要继承
Repository<T, ID>, CrudRepository<T, ID>或者JpaRepository<T, ID>接口即可。
此处,T是entity的泛型类,ID是该entity的主键类型。

  • 几个主要的Repository
public interface CrudRepository<T, ID> extends Repository<T, ID>;

public interface PagingAndSortingRepository<T, ID> extends CrudRepository<T, ID>;

public interface JpaRepository<T, ID> extends PagingAndSortingRepository<T, ID>;

注:
JpaRepository在CRUD之外提供了一些额外的功能:

  • flush()
    将所有没有与数据库同步的entity刷新到数据库中。
  • saveAndFlush()
    保存并立即flush。
  • deleteInBatch()
    批量删除。

 

3. 约定俗成的派生查询方法

Spring Data JPA虽然引入了一些系统的Repository,它们中的一些方法能满足大部分的应用开发需求。
但是在实际开发过程中,业余需求是远远多于通用操作的,那么Spring Data JPA是如何解决这些问题的呢?

答案是派生查询方法

派生查询方法是指开发者在继承某一种Repository之后,再自定义一些方法,通用一些约定俗成的方法名(eg: find...By,read...By,query...By,count...By,get...By),当符合这些命名规范以后,当其它类调用这些方法时,Spring Data会根据方法名自动生成相应的JPQL查询语句。
需要注意的是: 方法的参数名要与entity定义的属性一致。

超过一个参数的情况:

  • And连接方法参数名
    如果方法中以And连接参数名,则生成的SQL中以and连接条件。
eg: 
findAllByAvailableAndExpired(Boolean available, Boolean expired);


  • Or连接方法参数名
    如果方法中以Or连接参数名,则生成的SQL中以or连接条件。
eg: 
findAllByAvailableOrExpired(Boolean available, Boolean expired);

代码示例:

public interface CouponTemplateRepository
        extends JpaRepository<CouponTemplateEntity, Long> {

    CouponTemplateEntity findByName(String name);

    List<CouponTemplateEntity> findAllByAvailable(Boolean available);


    List<CouponTemplateEntity> findAllByAvailableAndExpired(
            Boolean available, Boolean expired
    );

    /**
     * <h2>根据 expired 标记查找模板记录</h2>
     * where expired = ...
     * */
    List<CouponTemplateEntity> findAllByExpired(Boolean expired);

    /**
     * 根据shop ID + 可用状态查询店铺有多少券模板
     */
    Integer countByShopIdAndAvailable(Long shopId, Boolean available);

    ...
}

其它方法关键字:

除了以上的And和Or方法关键字以外,Spring Data JPA也提供了下列关键字来进行辅助定义。

  • Like
    类似SQL中的like关键字。

  • Containing
    属性是否包含了参数值。

  • IgnoreCase
    在做值比较的时候忽略大小写。

eg:
findByNameContainingIgnoreCase
  • Between
    属性值是否在一个区间范围内。

  • LessThan/GreaterThan
    比较属性值和参数的大小。

 

4. JPQL定义Query查询

派生查询方法可以解决大部分简单查询需求,但是如果查询参数过多,就会导致方法签名过长,这种方法就不再合适,此时应当使用更轻量级的@Query功能。

Query注解可以支持JPQL和原生SQL。
使用Query注解的方式,开发者可以随意定义方法名,无需再遵循派生方法的命名规范。
只需要在Repository类中定义方法,并在方法上添加Query注解,再提供相应的JPQL或者原生SQL即可。

例子代码(JPQL语句)

public interface CouponTemplateRepository
        extends JpaRepository<CouponTemplateEntity, Long> {

    @Query("FROM CouponTemplate WHERE name = ?1")
    CouponTemplateEntity findByName(String name);

    @Query("FROM CouponTemplate WHERE available = ?1 AND expired = ?2")
    List<CouponTemplateEntity> findAllByAvailableAndExpired(Boolean available, Boolean expired);
}


例子代码(原生SQL语句)

public interface CouponTemplateRepository
        extends JpaRepository<CouponTemplateEntity, Long> {

    @Query("SELECT * FROM coupon_template WHERE name = :name")
    CouponTemplateEntity findByName(@Param("name")String name);

    @Query("SELECT * FROM coupon_template WHERE available = :available AND expired = :expired")
    List<CouponTemplateEntity> findAllByAvailableAndExpired(@Param("available")Boolean available, 
      @Param("expired")Boolean expired);
}

更新语句

那么对于UPDATE语句来说,应当如何实现数据修改呢?
答案还是使用@Query注解,但是需要添加一个额外的@Modifuing注解表明修改数据。

例子代码

public interface CouponTemplateRepository
        extends JpaRepository<CouponTemplateEntity, Long> {
    @Modifying
    @Query("update CouponTemplateEntity c set c.available = 0 where c.id = :id")
    int makeCouponUnavailable(@Param("id") Long id);
}

 

5. 命名查询(NamedQuery)

命名查询指的是为一段查询语句指定一个名称。

当我们执行这段语句的时候,只需通过这个名称就可以间接引用它对应的查询语句。
@NamedQuery注解支持JPQL语句,@NamedNativeQuery注解支持原生SQL语句。

例子代码

@Data
@NoArgsConstructor
@AllArgsConstructor
@Entity
@Builder
@EntityListeners(AuditingEntityListener.class)
@Table(name = "coupon_template")
@NamedQuery(name = "CouponTemplateEntity.findAllAvailable", query = "FROM CouponTemplateEntity where available=?1")
@NamedNativeQuery(name = "CouponTemplateEntity.findByShopId", query = "SELECT * FROM  coupon_template where shop_id=?", resultClass = CouponTemplateEntity.class)
public class CouponTemplateEntity implements Serializable {
    // 以下代码省略
}

使用命名查询

方法1(Java源代码方式)

Query q = em.createNamedQuery("CouponTemplateEntity.findAllAvailable");
q.setParameter(1, true);
List a = q.getResultList();

方法2(直接在Repository类中增加新的方法)

public interface CouponTemplateRepository
        extends JpaRepository<CouponTemplateEntity, Long> {
        List<CouponTemplate> findAllAvailable(Boolean available);
}

 

6. SpringBoot项目中使用JPA基本步骤

  1. 添加依赖
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
  1. application.yml中增加jpa相关配置
spring:
  jpa:
    show-sql: true
    hibernate:
      # 在生产环境全部为none,防止ddl结构被自动执行
      ddl-auto: none
    properties:
      hibernate.format_sql: true
      hibernate.show_sql: true
      hibernate.dialect: org.hibernate.dialect.MySQLDialect
    open-in-view: false
  1. 编写实体类,Repository注入到Service

有关SpringBoot JPA开发的更多相关文章

  1. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  2. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  3. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

  4. ruby - 在 Windows 机器上使用 Ruby 进行开发是否会适得其反? - 2

    这似乎非常适得其反,因为太多的gem会在window上破裂。我一直在处理很多mysql和ruby​​-mysqlgem问题(gem本身发生段错误,一个名为UnixSocket的类显然在Windows机器上不能正常工作,等等)。我只是在浪费时间吗?我应该转向不同的脚本语言吗? 最佳答案 我在Windows上使用Ruby的经验很少,但是当我开始使用Ruby时,我是在Windows上,我的总体印象是它不是Windows原生系统。因此,在主要使用Windows多年之后,开始使用Ruby促使我切换回原来的系统Unix,这次是Linux。Rub

  5. ruby-on-rails - 在 Rails 开发环境中为 .ogv 文件设置 Mime 类型 - 2

    我正在玩HTML5视频并且在ERB中有以下片段:mp4视频从在我的开发环境中运行的服务器很好地流式传输到chrome。然而firefox显示带有海报图像的视频播放器,但带有一个大X。问题似乎是mongrel不确定ogv扩展的mime类型,并且只返回text/plain,如curl所示:$curl-Ihttp://0.0.0.0:3000/pr6.ogvHTTP/1.1200OKConnection:closeDate:Mon,19Apr201012:33:50GMTLast-Modified:Sun,18Apr201012:46:07GMTContent-Type:text/plain

  6. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

  7. 【鸿蒙应用开发系列】- 获取系统设备信息以及版本API兼容调用方式 - 2

    在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList​()Obt

  8. 微信小程序开发入门与实战(Behaviors使用) - 2

    @作者:SYFStrive @博客首页:HomePage📜:微信小程序📌:个人社区(欢迎大佬们加入)👉:社区链接🔗📌:觉得文章不错可以点点关注👉:专栏连接🔗💃:感谢支持,学累了可以先看小段由小胖给大家带来的街舞👉微信小程序(🔥)目录自定义组件-behaviors    1、什么是behaviors    2、behaviors的工作方式    3、创建behavior    4、导入并使用behavior    5、behavior中所有可用的节点    6、同名字段的覆盖和组合规则总结最后自定义组件-behaviors    1、什么是behaviorsbehaviors是小程序中,用于实现

  9. ruby-on-rails - environment.rb 中设置的常量在开发模式中消失 - 2

    了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl

  10. ruby - Rails 开发服务器、PDFKit 和多线程 - 2

    我有一个使用PDFKit呈现网页的pdf版本的Rails应用程序。我使用Thin作为开发服务器。问题是当我处于开发模式时。当我使用“bundleexecrailss”启动我的服务器并尝试呈现任何PDF时,整个过程会陷入僵局,因为当您呈现PDF时,会向服务器请求一些额外的资源,如图像和css,看起来只有一个线程.如何配置Rails开发服务器以运行多个工作线程?非常感谢。 最佳答案 我找到的最简单的解决方案是unicorn.geminstallunicorn创建一个unicorn.conf:worker_processes3然后使用它:

随机推荐