詳解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傳參是否合理)。當(dāng)前端無查詢條件時則會導(dǎo)致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()方法源碼超級簡單,三行代碼搞定(其實(shí)嚴(yán)格來說就一行代碼):
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
 
而我的問題就是由第二點(diǎn)引起的,因為類型不相同┴─┴︵╰(‵□′╰)
教訓(xùn)總結(jié)
- 不建議使用Spring的StringUtils的isEmpty()對非String類型的對象判空。(這里建議換成apache common的StringUtils或者Google Guava的Strings,這兩個工具包都是類型強(qiáng)約束的)
 - 無論是修改哪處的代碼都最好檢查一下引用,避免修復(fù)小問題引出大問題
 
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:沒有了
欄 目:Java
下一篇:沒有了
本文標(biāo)題:詳解Spring的StringUtils踩坑記錄
本文地址:http://www.jygsgssxh.com/a1/Java/8692.html
您可能感興趣的文章
- 01-10Springboot中@Value的使用詳解
 - 01-10淺談Java中真的只有值傳遞么
 - 01-10如何解決線程太多導(dǎo)致java socket連接池出現(xiàn)的問題
 - 01-10springboot實(shí)現(xiàn)文件上傳步驟解析
 - 01-10java實(shí)現(xiàn)的順時針/逆時針打印矩陣操作示例
 - 01-10springboot jta atomikos實(shí)現(xiàn)分布式事物管理
 - 01-10java判斷是否空最簡單的方法
 - 01-10java郵件亂碼的徹底解決方案
 - 01-10JAVA8獨(dú)有的map遍歷方式(非常好用)
 - 01-10mybatis分頁絕對路徑寫法過程詳解
 


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


