PageHelper引發(fā)的幽靈數(shù)據(jù)問題解析
前言
最近測試反饋一個問題,某個查詢全量信息的接口,有時候返回全量數(shù)據(jù),符合預期,但是偶爾又只返回1條數(shù)據(jù),簡直就是“見鬼”了,究竟是為什么出現(xiàn)這樣的“幽靈數(shù)據(jù)”呢?
大膽猜測
首先我們看了下這對代碼的業(yè)務邏輯,非常的簡單,總共沒有幾行代碼,也沒有分頁邏輯,代碼如下:
public List<SdSubscription> findAll() {
return sdSubscriptionMapper.selectAll();
}
那么究竟是咋回事呢?講道理不可能出現(xiàn)這種情況的啊,不要慌,我們加點日志,將日志級別調整為DEBUG,讓日志飛一段時間。
public List<SdSubscription> findAll() {
log.info("find the sub start .....");
List<SdSubscription> subs = sdSubscriptionMapper.selectAll();
log.info("find the sub end .....");
return subs;
}
果不其然,日志中出現(xiàn)了奇奇怪怪的分頁參數(shù),如下圖所示:

果然是PageHelper這個開源框架搞的鬼,我想大家都用過吧,分頁非常方便,那么究竟為什么別人都沒問題,單單就我會出現(xiàn)問題呢?
PageHelper工作原理
為了回答上面的疑問,我們先看看PageHelper框架的工作原理吧。
PageHelper 是一個開源的 MyBatis 分頁插件,它可以幫助開發(fā)者在查詢數(shù)據(jù)時,快速的實現(xiàn)分頁功能。
PageHelper 的工作原理可以簡單概括為以下幾個步驟:
- 在需要進行分頁的查詢方法前,調用
PageHelper的靜態(tài)方法startPage(),設置當前頁碼和每頁顯示的記錄數(shù)。它會將分頁信息放到線程的ThreadLocal中,那么在線程的任何地方都可以訪問了。 - 當查詢方法執(zhí)行時,
PageHelper會自動攔截查詢語句,如果發(fā)現(xiàn)線程的ThreadLocal中有分頁信息,那么就會在其前后添加分頁語句,例如MySQL中的LIMIT語句。 - 查詢結果將被包裝在
Page對象中返回,該對象包含分頁信息和查詢結果列表。 - 在查詢方法執(zhí)行完畢后,會在
finally中清除線程ThreadLocal中的分頁信息,避免分頁設置對其他查詢方法的影響。
PageHelper 的實現(xiàn)原理主要依賴于攔截器技術和反射機制,通過攔截查詢語句并動態(tài)生成分頁語句,實現(xiàn)了簡單、高效、通用的分頁功能。具體源碼在下圖的類中,非常容易看懂。

明白了PageHelper的工作原理后,反復檢查代碼,都沒有調用過startPage,debug查看ThreadLocal中也沒有分頁信息啊,懵逼中。那我看看別人寫的添加分頁參數(shù)的代碼吧,不看不知道,一看嚇一跳。

原來有位“可愛”的同事竟然在查詢后,加了一個分頁,就是把分頁信息放到線程的ThreadLocal中。
那大家是不是有疑問,丁是丁,矛是矛,你的線程關我何事?這就要說到我們的tomcat了。
Tomcat請求流程
其實這就涉及到我們的tomcat相關知識了,我們一個瀏覽器發(fā)一個接口請求,經(jīng)過我們的tomcat的,究竟是一個什么樣的流程呢?
- 客戶端發(fā)送
HTTP請求到Tomcat服務器。 Tomcat的HTTP連接器(Connector)接收到請求,將連接請求交給線程池Executor處理,解析它,然后將請求轉發(fā)給對應的Web應用程序。Tomcat的Web應用程序容器(Container)接收到請求,根據(jù)請求的URL找到對應的Servlet。
關于tomcat中使用線程池提交瀏覽器的連接請求的源碼如下:

從而得知,你的連接請求是從線程池從拿的,而拿到的這個線程恰好是一個“臟線程”,在ThreadLocal中放了分頁信息,導致你這邊出現(xiàn)問題。
總結
后來追問了同事具體原因,才發(fā)現(xiàn)是粗心導致的。有些bug總是出現(xiàn)的莫名其妙,就像生活一樣。所以關鍵的是我們在使用一些開源框架的時候一定要掌握底層實現(xiàn)的原理、核心的機制,這樣才能夠在解決一些問題的時候有據(jù)可循。
以上就是PageHelper引發(fā)的幽靈數(shù)據(jù)問題解析的詳細內容,更多關于PageHelper幽靈數(shù)據(jù)的資料請關注腳本之家其它相關文章!
相關文章
Java Validation Api如何實現(xiàn)自定義注解
這篇文章主要介紹了Java Validation Api如何實現(xiàn)自定義注解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2020-09-09

