一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術|正則表達式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務器之家 - 編程語言 - Java教程 - Spring Data JPA實現動態查詢的兩種方法

Spring Data JPA實現動態查詢的兩種方法

2020-09-05 11:07lulll Java教程

本篇文章主要介紹了Spring Data JPA實現動態查詢的兩種方法,具有一定的參考價值,有興趣的可以了解一下。

前言

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

Criteria API

這套API可用于構建對數據庫的查詢。

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

如下即為元數據模型。創建一個元模型類,類名最后一個字符為下劃線,內部的成員變量與UserInfo.class這個實體類的屬性值相對應。

?
1
2
3
4
5
6
7
@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并不依賴具體的數據庫,可以根據數據庫類型的不同生成對應數據庫類型的SQL,所以其為可移植的。

面向對象。Criteria API是使用的是各種類和對象如CriteriaQuery、Predicate等構建查詢,是面向對象的。而如果直接書寫SQL則相對于面向的是字符串。

第一種:通過JPA的Criteria API實現

  1. EntityManager獲取CriteriaBuilder
  2. CriteriaBuilder創建CriteriaQuery
  3. CriteriaQuery指定要查詢的表,得到Root<UserInfo>,Root代表要查詢的表
  4. CriteriaBuilder創建條件Predicate,Predicate相對于SQL的where條件,多個Predicate可以進行與、或操作。
  5. 通過EntityManager創建TypedQuery
  6. TypedQuery執行查詢,返回結果
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
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層接口實現JpaSpecificationExecutor<T>接口

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

?
1
2
3
4
5
6
7
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實現JpaSpecificationExecutor

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

實現Specification

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
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.rar

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持服務器之家。

原文鏈接:http://www.jianshu.com/p/45ad65690e33#

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 天天综合网天天做天天受 | 国产卡一卡二卡3卡乱码免费 | 冰漪丰满大乳人体图片欣赏 | 黑白配高清hd在线视频 | 精品久久久久香蕉网 | 国产精品福利在线观看秒播 | 国产香蕉97碰碰久久人人 | 校花被拖到野外伦小说 | 亚洲人成高清毛片 | 日韩网站在线 | 明星乱亚洲 | 天堂成人在线观看 | 国内精品免费 | 四虎1515hhcom| 国产精品久久亚洲一区二区 | 大伊香蕉精品视频一区 | 韩国一级淫片特黄特刺激 | 欧美特级午夜一区二区三区 | 男人日女人的逼视频 | 亚洲国产免费观看视频 | 欧美成人tv | 国产高清日韩 | 国产在线麻豆波多野结衣 | 美女的隐私无遮挡的网页 | 99久久国产综合精品女不卡 | 沉沦艳妇杨幂肉体小说 | 欧美精品1区2区 | 午夜伦伦电影理论片费看 | ak福利午夜在线观看 | 五月丁香啪啪. | caoporen97免费公开视频 | 欧美精品一区二区在线观看播放 | 美女尿口羞羞视频 | 日韩欧美一区二区在线 | 日本不卡高清免费v日本 | 国产亚洲福利一区二区免费看 | 91制片厂果冻传媒首页 | 九九九精品视频 | 国产精品男人的天堂 | 国产成人精品系列在线观看 | 国产黄频在线观看高清免费 |