Spring Data JPA實(shí)現(xiàn)動(dòng)態(tài)查詢的兩種方法
前言
一般在寫(xiě)業(yè)務(wù)接口的過(guò)程中,很有可能需要實(shí)現(xiàn)可以動(dòng)態(tài)組合各種查詢條件的接口。如果我們根據(jù)一種查詢條件組合一個(gè)方法的做法來(lái)寫(xiě),那么將會(huì)有大量方法存在,繁瑣,維護(hù)起來(lái)相當(dāng)困難。想要實(shí)現(xiàn)動(dòng)態(tài)查詢,其實(shí)就是要實(shí)現(xiàn)拼接SQL語(yǔ)句。無(wú)論實(shí)現(xiàn)如何復(fù)雜,基本都是包括select的字段,from或者join的表,where或者h(yuǎn)aving的條件。在Spring Data JPA有兩種方法可以實(shí)現(xiàn)查詢條件的動(dòng)態(tài)查詢,兩種方法都用到了Criteria API。
Criteria API
這套API可用于構(gòu)建對(duì)數(shù)據(jù)庫(kù)的查詢。
類(lèi)型安全。通過(guò)定義元數(shù)據(jù)模型,在程序編譯階段就可以對(duì)類(lèi)型進(jìn)行檢查,不像SQL需要與Mysql進(jìn)行交互后才能發(fā)現(xiàn)類(lèi)型問(wèn)題。
如下即為元數(shù)據(jù)模型。創(chuàng)建一個(gè)元模型類(lèi),類(lèi)名最后一個(gè)字符為下劃線,內(nèi)部的成員變量與UserInfo.class這個(gè)實(shí)體類(lèi)的屬性值相對(duì)應(yīng)。
@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ù)庫(kù),可以根據(jù)數(shù)據(jù)庫(kù)類(lèi)型的不同生成對(duì)應(yīng)數(shù)據(jù)庫(kù)類(lèi)型的SQL,所以其為可移植的。
面向?qū)ο蟆?/strong>Criteria API是使用的是各種類(lèi)和對(duì)象如CriteriaQuery、Predicate等構(gòu)建查詢,是面向?qū)ο蟮?。而如果直接?shū)寫(xiě)SQL則相對(duì)于面向的是字符串。
第一種:通過(guò)JPA的Criteria API實(shí)現(xiàn)
- EntityManager獲取CriteriaBuilder
- CriteriaBuilder創(chuàng)建CriteriaQuery
- CriteriaQuery指定要查詢的表,得到Root<UserInfo>,Root代表要查詢的表
- CriteriaBuilder創(chuàng)建條件Predicate,Predicate相對(duì)于SQL的where條件,多個(gè)Predicate可以進(jìn)行與、或操作。
- 通過(guò)EntityManager創(chuàng)建TypedQuery
- 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層接口實(shí)現(xiàn)JpaSpecificationExecutor<T>接口
JpaSpecificationExecutor如下,方法參數(shù)Specification接口有一個(gè)方法toPredicate,返回值正好是Criteria API中的Predicate,而Predicate相對(duì)于SQL的where條件。與上一個(gè)方法相比,這種寫(xiě)法不需要指定查詢的表是哪一張,也不需要自己通過(guò)Criteria API實(shí)現(xiàn)排序和分頁(yè),只需要通過(guò)新建Pageable、Sort對(duì)象并傳參給findAll方法即可,簡(jiǎn)便一些。
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實(shí)現(xiàn)JpaSpecificationExecutor
public interface UserInfoDao extends PagingAndSortingRepository<UserInfo, String>, JpaSpecificationExecutor<UserInfo> {}
實(shí)現(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; } }; }
項(xiàng)目代碼:springdatajpademo_jb51.rar
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- spring data jpa使用詳解(推薦)
- Spring Data Jpa的四種查詢方式詳解
- Spring Data JPA實(shí)現(xiàn)動(dòng)態(tài)條件與范圍查詢實(shí)例代碼
- Spring Data JPA 實(shí)現(xiàn)多表關(guān)聯(lián)查詢的示例代碼
- Spring Data JPA 復(fù)雜/多條件組合分頁(yè)查詢
- Spring?Data?Jpa?復(fù)雜查詢方式總結(jié)(多表關(guān)聯(lián)及自定義分頁(yè))
- Spring Data JPA調(diào)用存儲(chǔ)過(guò)程實(shí)例代碼
- Spring Data JPA的作用和用法小結(jié)
相關(guān)文章
詳解Spring 中如何控制2個(gè)bean中的初始化順序
本篇文章主要介紹了Spring 中如何控制2個(gè)bean中的初始化順序,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-10-10Spring Validation方法實(shí)現(xiàn)原理分析
這篇文章主要介紹了Spring Validation實(shí)現(xiàn)原理分析,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-07-07Java堆&優(yōu)先級(jí)隊(duì)列示例講解(下)
這篇文章主要通過(guò)示例詳細(xì)為大家介紹Java中的堆以及優(yōu)先級(jí)隊(duì)列,文中的示例代碼講解詳細(xì),對(duì)我們了解java有一定幫助,需要的可以參考一下2022-03-03如何基于sqlite實(shí)現(xiàn)kafka延時(shí)消息詳解
這篇文章主要給大家介紹了關(guān)于如何基于sqlite實(shí)現(xiàn)kafka延時(shí)消息的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2022-01-01SpringBoot整合JDBC的實(shí)現(xiàn)
這篇文章主要介紹了SpringBoot整合JDBC的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01使用jenkins部署springboot項(xiàng)目的方法步驟
這篇文章主要介紹了使用jenkins部署springboot項(xiàng)目的方法步驟,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-04-04