Spring Data JPA使用JPQL與原生SQL進(jìn)行查詢的操作
1、使用JPQL語句進(jìn)行查詢
JPQL語言(Java Persistence Query Language)是一種和SQL非常類似的中間性和對象化查詢語言,它最終會被編譯成針對不同底層數(shù)據(jù)庫的SQL語言,從而屏蔽不同數(shù)據(jù)庫的差異。
JPQL語言通過Query接口封裝執(zhí)行,Query 接口封裝了執(zhí)行數(shù)據(jù)庫查詢的相關(guān)方法。調(diào)用 EntityManager 的 Query、NamedQuery 及 NativeQuery 方法可以獲得查詢對象,進(jìn)而可調(diào)用Query接口的相關(guān)方法來執(zhí)行查詢操作。
JPQL是面向?qū)ο筮M(jìn)行查詢的語言,可以通過自定義的JPQL完成UPDATE和DELETE操作。JPQL不支持使用INSERT。對于UPDATE或DELETE操作,必須使用注解 @Modifying 進(jìn)行修飾。
【示例】使用JPQL語言進(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ù)庫訪問接口
* 使用JPQL語言
* @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語句進(jìn)行查詢
在使用原生SQL查詢時,也使用@Query注解。此時,nativeQuery參數(shù)需要設(shè)置為true。
【示例】使用原生SQL語句進(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ù)庫訪問接口測試類
* 使用原生SQL語言
* @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這兩個注解一起聲明,可以定義個性化更新操作。
Spring data jpa@query使用原生SQl,需要注意的坑
根據(jù)代碼來解說:
@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、返回的是一個page<Object[]>,數(shù)組中只保存了數(shù)據(jù),沒有對應(yīng)的key,只能根據(jù)返回數(shù)據(jù)的順序,依次注入到DTO中。
3、對于使用分頁,需要:“ORDER BY ?#{#pageable}”,可以直接傳入一個pageable對象,會自動解析。
4、注意格式問題,很多時候就是換行的時候,沒有空格。
5、仔細(xì)對應(yīng)數(shù)據(jù)庫中表字段,很多時候報某個字段找不到,就是因?yàn)樽侄蚊麑戝e,和數(shù)據(jù)庫中對應(yīng)不上。
6、這是解決使用微服務(wù),大量的數(shù)據(jù)都需要遠(yuǎn)程調(diào)用,會降低程序的性能。
7、使用Pageabel作為參數(shù)的時候,去進(jìn)行分頁。剛開始的時候,覺得還是一個可行的辦法,但是得注意的時候,當(dāng)需要排序的時候,是無法加入sort字段的。 會一直報錯left*。
8、針對7的解決方案,把原生SQL的數(shù)據(jù)查詢和countQuery分成兩個查詢方法。得到count,然后進(jìn)行判斷,若是等于0,則直接返回空集合;反之,則取獲取數(shù)據(jù)。 需要自己進(jìn)行分頁計算,傳入正確的pageNumber和pageSize。 大部分系統(tǒng)都是按照修改時間進(jìn)行降序排序。 所以,order by可以寫死。然后pageNumber和pageSize動態(tài)傳入。 pageNumber的算法= (pageNumber - 1) * pageSize, 前提是PageNumber是從1開始,若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);
}
以上為個人經(jīng)驗(yàn),希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot?如何通過?Profile?實(shí)現(xiàn)不同環(huán)境下的配置切換
SpringBoot通過profile實(shí)現(xiàn)在不同環(huán)境下的配置切換,比如常見的開發(fā)環(huán)境、測試環(huán)境、生產(chǎn)環(huán)境,SpringBoot常用配置文件主要有?2?種:properties?文件和yml文件,本文給大家詳細(xì)介紹SpringBoot?通過?Profile?實(shí)現(xiàn)不同環(huán)境下的配置切換,感興趣的朋友一起看看吧2022-08-08
解決執(zhí)行Junit單元測試報錯java.lang.ClassNotFoundException問題
這篇文章主要介紹了解決執(zhí)行Junit單元測試報錯java.lang.ClassNotFoundException問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-11-11
spring-boot 如何實(shí)現(xiàn)單次執(zhí)行程序
這篇文章主要介紹了spring-boot 實(shí)現(xiàn)單次執(zhí)行程序方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-09-09
mybatis返回map類型數(shù)據(jù)空值字段不顯示的解決方案
這篇文章主要介紹了mybatis返回map類型數(shù)據(jù)空值字段不顯示的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03
java web支持jsonp的實(shí)現(xiàn)代碼
這篇文章主要介紹了java web支持jsonp的實(shí)現(xiàn)代碼,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-11-11
IntelliJ IDEA 2020.1 EAP4 發(fā)布,重命名/更改簽名新功能一覽
這篇文章主要介紹了IntelliJ IDEA 2020.1 EAP4 發(fā)布,重命名/更改簽名新功能,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-04-04
Java基礎(chǔ)之多線程方法狀態(tài)和創(chuàng)建方法
Java中可以通過Thread類和Runnable接口來創(chuàng)建多個線程,下面這篇文章主要給大家介紹了關(guān)于Java基礎(chǔ)之多線程方法狀態(tài)和創(chuàng)建方法的相關(guān)資料,需要的朋友可以參考下2021-09-09

