詳解Spring的StringUtils踩坑記錄
起因
最近在寫CRUD的時候,發(fā)現(xiàn)有個分頁的VO寫的健壯性比較差,一時手癢改了一下,沒想到改了之后好幾個功能都出現(xiàn)了問題。
原VO關(guān)鍵代碼如下:
public class PageVo implements Serializable{ // ...省略所有無關(guān)代碼 Map<String, String> query }
這個VO是用于從前端分頁查詢時傳參,而query是用于傳遞查詢條件的(這里不討論用Map傳參是否合理)。當前端無查詢條件時則會導致query為null,如果不注意容易出現(xiàn)NPE。
所以我就改造成下面這樣了。
public class PageVo implements Serializable{ // ...省略所有無關(guān)代碼 Map<String, String> query=new HashMap<> }
但是沒想到就是這么簡單的改造居然都翻車(・ε・`)
沒辦法,只好去排查問題。
發(fā)現(xiàn)問題
想過很多種原因,但是我真沒想到居然是因為這樣(/‵Д′)/~ ╧╧,不多說了,問題關(guān)鍵代碼如下:
if (StringUtils.isEmpty(page.getQuery())) { // 省略處理邏輯 }
居然用StringUtils去判斷一個Map是否為空,好歹也換個CollectionUtils啊(╬ ̄皿 ̄)凸
雖然是前人挖坑,但是為什么Spring的`StringUtils居然設(shè)計成支持Object入?yún)⒛豲_o ....
想了一下,還是去看看源碼吧
源碼分析
StringUtils的isEmpty()方法源碼超級簡單,三行代碼搞定(其實嚴格來說就一行代碼):
public static boolean isEmpty(@Nullable Object str) { return (str == null || "".equals(str)); }
既然我的Map對象不為null,那么問題應(yīng)該是因為String的equals()方法。不多說,繼續(xù)跟蹤源碼
public boolean equals(Object anObject) { if (this == anObject) { return true; } // 問題出在這里 if (anObject instanceof String) { String anotherString = (String)anObject; int n = value.length; if (n == anotherString.value.length) { char v1[] = value; char v2[] = anotherString.value; int i = 0; while (n-- != 0) { if (v1[i] != v2[i]) return false; i++; } return true; } } return false; }
這個equals()方法的邏輯很簡單
- 入?yún)閚ull則返回true
- 入?yún)⒉粸镾tring類型返回false
- 入?yún)ο蠛蛅his對象都為String就比較它們內(nèi)置的char[]數(shù)組長度和每個char元素是否相同,滿足則返回true,否則返回false
而我的問題就是由第二點引起的,因為類型不相同┴─┴︵╰(‵□′╰)
教訓總結(jié)
- 不建議使用Spring的StringUtils的isEmpty()對非String類型的對象判空。(這里建議換成apache common的StringUtils或者Google Guava的Strings,這兩個工具包都是類型強約束的)
- 無論是修改哪處的代碼都最好檢查一下引用,避免修復小問題引出大問題
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持我們。
您可能感興趣的文章
- 01-10Springboot中@Value的使用詳解
- 01-10淺談Java中真的只有值傳遞么
- 01-10如何解決線程太多導致java socket連接池出現(xiàn)的問題
- 01-10springboot實現(xiàn)文件上傳步驟解析
- 01-10java實現(xiàn)的順時針/逆時針打印矩陣操作示例
- 01-10springboot jta atomikos實現(xiàn)分布式事物管理
- 01-10java判斷是否空最簡單的方法
- 01-10java郵件亂碼的徹底解決方案
- 01-10JAVA8獨有的map遍歷方式(非常好用)
- 01-10mybatis分頁絕對路徑寫法過程詳解


閱讀排行
本欄相關(guān)
- 01-10Java實現(xiàn)動態(tài)模擬時鐘
- 01-10Springboot中@Value的使用詳解
- 01-10JavaWeb實現(xiàn)郵件發(fā)送功能
- 01-10利用Java實現(xiàn)復制Excel工作表功能
- 01-10Java實現(xiàn)動態(tài)數(shù)字時鐘
- 01-10java基于poi導出excel透視表代碼實例
- 01-10java實現(xiàn)液晶數(shù)字字體顯示當前時間
- 01-10基于Java驗證jwt token代碼實例
- 01-10Java動態(tài)顯示當前日期和時間
- 01-10淺談Java中真的只有值傳遞么
隨機閱讀
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10C#中split用法實例總結(jié)
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10delphi制作wav文件的方法
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-11ajax實現(xiàn)頁面的局部加載
- 04-02jquery與jsp,用jquery
- 08-05織夢dedecms什么時候用欄目交叉功能?
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-10使用C語言求解撲克牌的順子及n個骰子