Spring Data JPA使用JPQL與原生SQL進(jìn)行查詢的操作
1、使用JPQL語(yǔ)句進(jìn)行查詢
JPQL語(yǔ)言(Java Persistence Query Language)是一種和SQL非常類似的中間性和對(duì)象化查詢語(yǔ)言,它最終會(huì)被編譯成針對(duì)不同底層數(shù)據(jù)庫(kù)的SQL語(yǔ)言,從而屏蔽不同數(shù)據(jù)庫(kù)的差異。
JPQL語(yǔ)言通過(guò)Query接口封裝執(zhí)行,Query 接口封裝了執(zhí)行數(shù)據(jù)庫(kù)查詢的相關(guān)方法。調(diào)用 EntityManager 的 Query、NamedQuery 及 NativeQuery 方法可以獲得查詢對(duì)象,進(jìn)而可調(diào)用Query接口的相關(guān)方法來(lái)執(zhí)行查詢操作。
JPQL是面向?qū)ο筮M(jìn)行查詢的語(yǔ)言,可以通過(guò)自定義的JPQL完成UPDATE和DELETE操作。JPQL不支持使用INSERT。對(duì)于UPDATE或DELETE操作,必須使用注解 @Modifying 進(jìn)行修飾。
【示例】使用JPQL語(yǔ)言進(jìn)行查詢
package com.pjb.jpauserdemo.dao; import com.pjb.jpauserdemo.entity.UserInfo; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.data.jpa.repository.Modifying; import org.springframework.data.jpa.repository.Query; import org.springframework.data.repository.query.Param; import org.springframework.stereotype.Repository; import java.util.List; /** * 用戶信息數(shù)據(jù)庫(kù)訪問(wèn)接口 * 使用JPQL語(yǔ)言 * @author pan_junbiao **/ @Repository public interface UserJpqlDao extends JpaRepository<UserInfo,Integer> { /** * 根據(jù)用戶姓名,查詢用戶信息 */ @Query("SELECT u FROM UserInfo u WHERE u.userName = ?1") public UserInfo getUserInfoByName(String name); /** * 根據(jù)用戶姓名,模糊查詢用戶列表 */ @Query("SELECT u FROM UserInfo u WHERE u.userName like %:name%") public List<UserInfo> getUserListByName(String name); /** * 修改用戶姓名 */ @Modifying @Query("UPDATE UserInfo u SET u.userName = :name WHERE u.userId = :id") public int updateUserName(@Param("id")int userId, @Param("name")String userName); }
2、使用原生SQL語(yǔ)句進(jìn)行查詢
在使用原生SQL查詢時(shí),也使用@Query注解。此時(shí),nativeQuery參數(shù)需要設(shè)置為true。
【示例】使用原生SQL語(yǔ)句進(jìn)行查詢
package com.pjb.jpauserdemo.dao; import com.pjb.jpauserdemo.entity.UserInfo; import org.springframework.data.jpa.repository.JpaRepository; import org.springframework.data.jpa.repository.Modifying; import org.springframework.data.jpa.repository.Query; import org.springframework.data.repository.query.Param; import org.springframework.stereotype.Repository; import org.springframework.transaction.annotation.Transactional; import java.util.List; /** * 用戶信息數(shù)據(jù)庫(kù)訪問(wèn)接口測(cè)試類 * 使用原生SQL語(yǔ)言 * @author pan_junbiao **/ @Repository public interface UserSqlDao extends JpaRepository<UserInfo,Integer> { /** * 根據(jù)用戶ID,獲取用戶信息 */ @Query(value = "SELECT * FROM tb_user WHERE user_id = :id",nativeQuery = true) public UserInfo getUserById(@Param("id")int userId); /** * 根據(jù)用戶姓名,模糊查詢用戶列表 */ @Query(value = "SELECT * FROM tb_user WHERE user_name LIKE %:userName%",nativeQuery = true) public List<UserInfo> getUserListByName(@Param("userName")String userName); /** * 修改用戶姓名 */ @Modifying @Query(value = "UPDATE tb_user SET user_name = :name WHERE user_id = :id",nativeQuery = true) public int updateUserName(@Param("id")int userId, @Param("name")String userName); }
可以看到,@Query與@Modifying這兩個(gè)注解一起聲明,可以定義個(gè)性化更新操作。
Spring data jpa@query使用原生SQl,需要注意的坑
根據(jù)代碼來(lái)解說(shuō):
@Query(value = "select bill.id_ as id, bill.created_date as date, bill.no, lawyer_case .case_no as caseNo, " + "lawyer_case .case_name as caseName, customer.no as customerNo, customer.cn_name as customerName, " + "bill.total_expense_after_tax, bill.collected_money, bill.book_ticket_amount, bill.version " + "e1.name as creator, bill.status" + "from bill " + "left join lawyer_case on lawyer_case .case_no=bill.case_no " + "left join customer on customer.no=bill.customer_no " + "left join employee e1 on e1.id_=bill.creator " + "where IF (?1!='', customer_no=?1, 1=1) " + "and IF (?2!='', case_no=?2, 1=1) " + "and IF (?3!='', status=?3, 1=1) " + "and IF (?4!='', creator'%',?4,'%')), 1=1) " + "and create_by=?5 " + "ORDER BY ?#{#pageable} ", countQuery = "select count(*) " + "from bill " + "left join lawyer_case on lawyer_case .case_no=bill.case_no " + "left join customer on customer.no=bill.customer_no " + "left join employee e1 on e1.id_=bill.creator " + "where IF (?1!='', customer_no=?1, 1=1) " + "and IF (?2!='', case_no=?2, 1=1) " + "and IF (?3!='', status=?3, 1=1) " + "and IF (?4!='', creator'%',?4,'%')), 1=1) " + "and create_by=?5 "+ "ORDER BY ?#{#pageable} ", nativeQuery = true) Page<Object[]> findAllBill(String customerNo, String caseNo, Integer status, String creator, String createBy, Pageable pageable);
需要注意的方法有以下幾點(diǎn):
1、From 不支持重命名.
2、返回的是一個(gè)page<Object[]>,數(shù)組中只保存了數(shù)據(jù),沒(méi)有對(duì)應(yīng)的key,只能根據(jù)返回?cái)?shù)據(jù)的順序,依次注入到DTO中。
3、對(duì)于使用分頁(yè),需要:“ORDER BY ?#{#pageable}”,可以直接傳入一個(gè)pageable對(duì)象,會(huì)自動(dòng)解析。
4、注意格式問(wèn)題,很多時(shí)候就是換行的時(shí)候,沒(méi)有空格。
5、仔細(xì)對(duì)應(yīng)數(shù)據(jù)庫(kù)中表字段,很多時(shí)候報(bào)某個(gè)字段找不到,就是因?yàn)樽侄蚊麑戝e(cuò),和數(shù)據(jù)庫(kù)中對(duì)應(yīng)不上。
6、這是解決使用微服務(wù),大量的數(shù)據(jù)都需要遠(yuǎn)程調(diào)用,會(huì)降低程序的性能。
7、使用Pageabel作為參數(shù)的時(shí)候,去進(jìn)行分頁(yè)。剛開(kāi)始的時(shí)候,覺(jué)得還是一個(gè)可行的辦法,但是得注意的時(shí)候,當(dāng)需要排序的時(shí)候,是無(wú)法加入sort字段的。 會(huì)一直報(bào)錯(cuò)left*。
8、針對(duì)7的解決方案,把原生SQL的數(shù)據(jù)查詢和countQuery分成兩個(gè)查詢方法。得到count,然后進(jìn)行判斷,若是等于0,則直接返回空集合;反之,則取獲取數(shù)據(jù)。 需要自己進(jìn)行分頁(yè)計(jì)算,傳入正確的pageNumber和pageSize。 大部分系統(tǒng)都是按照修改時(shí)間進(jìn)行降序排序。 所以,order by可以寫死。然后pageNumber和pageSize動(dòng)態(tài)傳入。 pageNumber的算法= (pageNumber - 1) * pageSize, 前提是PageNumber是從1開(kāi)始,若0,則pageNumber=pageNumber * PageSize; 這樣就可以保證數(shù)據(jù)的正確。
/** * pageInfos: 轉(zhuǎn)換之后的數(shù)據(jù)。 * pageable:傳入的pageable. * totalPage: 第一條SQL算好的返回值。 * 這樣就可以統(tǒng)一的返回各種pageDTO。 */ private Page<T> convertForPage(List<T> pageInfos, Pageable pageable, Integer totalPage) { return new PageImpl<>(pageInfos, pageable, totalPage); }
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
如何使用Java?8中DateTimeFormatter類型轉(zhuǎn)換日期格式詳解
這篇文章主要介紹了如何使用Java?8中DateTimeFormatter類型轉(zhuǎn)換日期格式詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07Java通過(guò)SSLEngine與NIO實(shí)現(xiàn)HTTPS訪問(wèn)的操作方法
這篇文章主要介紹了Java通過(guò)SSLEngine與NIO實(shí)現(xiàn)HTTPS訪問(wèn),需要在Connect操作、Connected操作、Read和Write操作中加入SSL相關(guān)的處理即可,需要的朋友可以參考下2021-08-08maven中心倉(cāng)庫(kù)OSSRH使用簡(jiǎn)介(推薦)
這篇文章主要介紹了maven中心倉(cāng)庫(kù)OSSRH使用簡(jiǎn)介,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04java中對(duì)字符串每個(gè)字符統(tǒng)計(jì)的方法
java中對(duì)字符串每個(gè)字符統(tǒng)計(jì)的方法,需要的朋友可以參考一下2013-03-03使用maven如何將項(xiàng)目中的test代碼打包進(jìn)jar中
這篇文章主要介紹了使用maven如何將項(xiàng)目中的test代碼打包進(jìn)jar中,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03SpringBoot中yml多環(huán)境配置的3種方法
這篇文章主要給大家介紹了SpringBoot中yml多環(huán)境配置的3種方法,文中有詳細(xì)的代碼示例供大家參考,對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-10-10SpringCloud遠(yuǎn)程服務(wù)調(diào)用三種方式及原理
本文給大家介紹SpringCloud遠(yuǎn)程服務(wù)調(diào)用實(shí)戰(zhàn)筆記,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2022-12-12SpringBoot動(dòng)態(tài)數(shù)據(jù)源連接測(cè)試的操作詳解
這篇文章主要介紹了SpringBoot動(dòng)態(tài)數(shù)據(jù)源連接測(cè)試的操作步驟,文中通過(guò)代碼示例和圖文結(jié)合的方式給大家講解的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-03-03