聊聊Spring data jpa @query使用原生SQl,需要注意的坑
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、返回的是一個(gè)page<Object[]>,數(shù)組中只保存了數(shù)據(jù),沒有對(duì)應(yīng)的key,只能根據(jù)返回?cái)?shù)據(jù)的順序,依次注入到DTO中。
- 3、對(duì)于使用分頁,需要:“ORDER BY ?#{#pageable}”,可以直接傳入一個(gè)pageable對(duì)象,會(huì)自動(dòng)解析。
- 4、注意格式問題,很多時(shí)候就是換行的時(shí)候,沒有空格。
- 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)行分頁。剛開始的時(shí)候,覺得還是一個(gè)可行的辦法,但是得注意的時(shí)候,當(dāng)需要排序的時(shí)候,是無法加入sort字段的。 會(huì)一直報(bào)錯(cuò)left*。
- 8、針對(duì)7的解決方案,把原生SQL的數(shù)據(jù)查詢和countQuery分成兩個(gè)查詢方法。得到count,然后進(jìn)行判斷,若是等于0,則直接返回空集合;反之,則取獲取數(shù)據(jù)。 需要自己進(jìn)行分頁計(jì)算,傳入正確的pageNumber和pageSize。
大部分系統(tǒng)都是按照修改時(shí)間進(jìn)行降序排序。 所以,order by可以寫死。然后pageNumber和pageSize動(dòng)態(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); }
SpringData JPA @Query動(dòng)態(tài)SQL語句
這次有個(gè)需求,需要?jiǎng)討B(tài)的sql語句去查詢,但是@Query正常情況下SQL語句是寫死的,在查找了很多資料后,想到了一個(gè)好的解決辦法
思路
利用MYSQL的判斷來拼接SQL語句
實(shí)現(xiàn)
先上代碼
@Query(value = "select * from project_demand where project_id=?1 and if(?2!='',demand_id in (select demand_id from demand_user where user_id=?2),1=1)",nativeQuery = true) List<ProjectDemand> getListByUser(String projectId,String userId);
紅色部分,就是生成動(dòng)態(tài)SQL的方法,利用MYSQL的if函數(shù)和我們傳遞的參數(shù)去進(jìn)行判斷,然后獲取SQL語句。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
淺析java中String類型中“==”與“equal”的區(qū)別
這篇文章主要介紹了淺析java中String類型中“==”與“equal”的區(qū)別,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-08-08java編譯時(shí)與運(yùn)行時(shí)概念與實(shí)例詳解
本篇文章通過實(shí)例對(duì) java程序編譯時(shí)與運(yùn)行時(shí)進(jìn)行了詳解,需要的朋友可以參考下2017-04-04springboot中使用Hibernate-Validation校驗(yàn)參數(shù)詳解
這篇文章主要為大家介紹了springboot中使用Hibernate-Validation校驗(yàn)參數(shù)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07Spring?Boot整合郵箱發(fā)送郵件實(shí)例
大家好,本篇文章主要講的是Spring?Boot整合郵箱發(fā)送郵件實(shí)例,感興趣的同學(xué)趕快來看一看吧,對(duì)你有幫助的話記得收藏一下2022-02-02普通對(duì)象使用spring容器中的對(duì)象的實(shí)現(xiàn)方法
這篇文章主要介紹了普通對(duì)象使用spring容器中的對(duì)象的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06SpringCache 分布式緩存的實(shí)現(xiàn)方法(規(guī)避redis解鎖的問題)
這篇文章主要介紹了SpringCache 分布式緩存的實(shí)現(xiàn)方法(規(guī)避redis解鎖的問題),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11