欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring Data JPA實現(xiàn)動態(tài)查詢的兩種方法

 更新時間:2017年04月01日 11:21:13   作者:lulll  
本篇文章主要介紹了Spring Data JPA實現(xiàn)動態(tài)查詢的兩種方法,具有一定的參考價值,有興趣的可以了解一下。

前言

一般在寫業(yè)務接口的過程中,很有可能需要實現(xiàn)可以動態(tài)組合各種查詢條件的接口。如果我們根據(jù)一種查詢條件組合一個方法的做法來寫,那么將會有大量方法存在,繁瑣,維護起來相當困難。想要實現(xiàn)動態(tài)查詢,其實就是要實現(xiàn)拼接SQL語句。無論實現(xiàn)如何復雜,基本都是包括select的字段,from或者join的表,where或者having的條件。在Spring Data JPA有兩種方法可以實現(xiàn)查詢條件的動態(tài)查詢,兩種方法都用到了Criteria API。

Criteria API

這套API可用于構(gòu)建對數(shù)據(jù)庫的查詢。

類型安全。通過定義元數(shù)據(jù)模型,在程序編譯階段就可以對類型進行檢查,不像SQL需要與Mysql進行交互后才能發(fā)現(xiàn)類型問題。

如下即為元數(shù)據(jù)模型。創(chuàng)建一個元模型類,類名最后一個字符為下劃線,內(nèi)部的成員變量與UserInfo.class這個實體類的屬性值相對應。

@StaticMetamodel(UserInfo.class)
public class UserInfo_ {
  public static volatile SingularAttribute<UserInfo, Integer> userId;
  public static volatile SingularAttribute<UserInfo, String> name;
  public static volatile SingularAttribute<UserInfo, Integer> age;
  public static volatile SingularAttribute<UserInfo, Long> high;
}

可移植。API并不依賴具體的數(shù)據(jù)庫,可以根據(jù)數(shù)據(jù)庫類型的不同生成對應數(shù)據(jù)庫類型的SQL,所以其為可移植的。

面向?qū)ο蟆?/strong>Criteria API是使用的是各種類和對象如CriteriaQuery、Predicate等構(gòu)建查詢,是面向?qū)ο蟮?。而如果直接書寫SQL則相對于面向的是字符串。

第一種:通過JPA的Criteria API實現(xiàn)

  1. EntityManager獲取CriteriaBuilder
  2. CriteriaBuilder創(chuàng)建CriteriaQuery
  3. CriteriaQuery指定要查詢的表,得到Root<UserInfo>,Root代表要查詢的表
  4. CriteriaBuilder創(chuàng)建條件Predicate,Predicate相對于SQL的where條件,多個Predicate可以進行與、或操作。
  5. 通過EntityManager創(chuàng)建TypedQuery
  6. TypedQuery執(zhí)行查詢,返回結(jié)果
public class UserInfoExtendDao {

 @PersistenceContext(unitName = "springJpa")
 EntityManager em;

 public List<UserInfo> getUserInfo(String name,int age,int high) {
   CriteriaBuilder cb = em.getCriteriaBuilder();
   CriteriaQuery<UserInfo> query = cb.createQuery(UserInfo.class);

   //from
   Root<UserInfo> root = query.from(UserInfo.class);

   //where
   Predicate p1 = null;
   if(name!=null) {
     Predicate p2 = cb.equal(root.get(UserInfo_.name),name);
     if(p1 != null) {
       p1 = cb.and(p1,p2);
     } else {
       p1 = p2;
     }
   }

   if(age!=0) {
     Predicate p2 = cb.equal(root.get(UserInfo_.age), age);
     if(p1 != null) {
       p1 = cb.and(p1,p2);
     } else {
       p1 = p2;
     }
   }

   if(high!=0) {
     Predicate p2 = cb.equal(root.get(UserInfo_.high), high);
     if(p1 != null) {
       p1 = cb.and(p1,p2);
     } else {
       p1 = p2;
     }
   }
   query.where(p1);

   List<UserInfo> userInfos = em.createQuery(query).getResultList();
   return userInfos;
 }
}

第二種:DAO層接口實現(xiàn)JpaSpecificationExecutor<T>接口

JpaSpecificationExecutor如下,方法參數(shù)Specification接口有一個方法toPredicate,返回值正好是Criteria API中的Predicate,而Predicate相對于SQL的where條件。與上一個方法相比,這種寫法不需要指定查詢的表是哪一張,也不需要自己通過Criteria API實現(xiàn)排序和分頁,只需要通過新建Pageable、Sort對象并傳參給findAll方法即可,簡便一些。

public interface JpaSpecificationExecutor<T> {
 T findOne(Specification<T> spec);
 List<T> findAll(Specification<T> spec);
 Page<T> findAll(Specification<T> spec, Pageable pageable);
 List<T> findAll(Specification<T> spec, Sort sort);
 long count(Specification<T> spec);
}

UserInfoDao實現(xiàn)JpaSpecificationExecutor

public interface UserInfoDao 
  extends PagingAndSortingRepository<UserInfo, String>, JpaSpecificationExecutor<UserInfo> {}

實現(xiàn)Specification

public static Specification<UserInfo> getSpec(final String name,final int age,final int high) {
   return new Specification<UserInfo>() {
     @Override
     public Predicate toPredicate(Root<UserInfo> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
       Predicate p1 = null;
       if(name!=null) {
         Predicate p2 = cb.equal(root.get(UserInfo_.name),name);
         if(p1 != null) {
           p1 = cb.and(p1,p2);
         } else {
           p1 = p2;
         }
       }

       if(age!=0) {
         Predicate p2 = cb.equal(root.get(UserInfo_.age), age);
         if(p1 != null) {
           p1 = cb.and(p1,p2);
         } else {
           p1 = p2;
         }
       }

       if(high!=0) {
         Predicate p2 = cb.equal(root.get(UserInfo_.high), high);
         if(p1 != null) {
           p1 = cb.and(p1,p2);
         } else {
           p1 = p2;
         }
       }

       return p1;
     }
   };
 }

項目代碼:springdatajpademo_jb51.rar

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關文章

  • springboot中如何替換class文件

    springboot中如何替換class文件

    這篇文章主要介紹了springboot中如何替換class文件,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • 詳解Spring 中如何控制2個bean中的初始化順序

    詳解Spring 中如何控制2個bean中的初始化順序

    本篇文章主要介紹了Spring 中如何控制2個bean中的初始化順序,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • SpringBoot增量/瘦身部署jar包的方式

    SpringBoot增量/瘦身部署jar包的方式

    SpringBoot 項目的部署一般采用全量jar 包方式部署相關項目,如果我們對相關的Contrller層進行相關業(yè)務調(diào)整就需要重新編譯全量jar 包太麻煩了,所以本文給大家介紹了使用SpringBoot 的增量/瘦身部署方式,需要的朋友可以參考下
    2024-01-01
  • Spring Validation方法實現(xiàn)原理分析

    Spring Validation方法實現(xiàn)原理分析

    這篇文章主要介紹了Spring Validation實現(xiàn)原理分析,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-07-07
  • Java堆&優(yōu)先級隊列示例講解(下)

    Java堆&優(yōu)先級隊列示例講解(下)

    這篇文章主要通過示例詳細為大家介紹Java中的堆以及優(yōu)先級隊列,文中的示例代碼講解詳細,對我們了解java有一定幫助,需要的可以參考一下
    2022-03-03
  • 淺談SpringCloud之zuul源碼解析

    淺談SpringCloud之zuul源碼解析

    這篇文章主要介紹了淺談SpringCloud之zuul源碼解析,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-02-02
  • 如何基于sqlite實現(xiàn)kafka延時消息詳解

    如何基于sqlite實現(xiàn)kafka延時消息詳解

    這篇文章主要給大家介紹了關于如何基于sqlite實現(xiàn)kafka延時消息的相關資料,文中通過實例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2022-01-01
  • SpringBoot整合JDBC的實現(xiàn)

    SpringBoot整合JDBC的實現(xiàn)

    這篇文章主要介紹了SpringBoot整合JDBC的實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-01-01
  • Java super和this的對比及使用

    Java super和this的對比及使用

    這篇文章主要介紹了Java super和this的對比及使用的相關資料,java中this與super會經(jīng)常在使用的時候混淆,需要的朋友可以參考下
    2017-08-08
  • 使用jenkins部署springboot項目的方法步驟

    使用jenkins部署springboot項目的方法步驟

    這篇文章主要介紹了使用jenkins部署springboot項目的方法步驟,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-04-04

最新評論