SpringBoot JPA开发

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
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,490评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,581评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,830评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,957评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,974评论 6 393
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,754评论 1 307
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,464评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,357评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,847评论 1 317
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,995评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,137评论 1 351
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,819评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,482评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,023评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,149评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,409评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,086评论 2 355

推荐阅读更多精彩内容