我天!xx.equals(null) 是什麼騷操做??

問題背景

個人天,最近作 Code Review 看到一個同事的騷操做,他寫了一個工具類,大概是這樣的:java

public static boolean isNull(Object object){
    return null == object || object.equals(null);
}

判斷空,通常不是 null == object 就夠了,object.equals(null) 是什麼騷操做?面試

寫程序這麼多年,第一次看這樣的寫法,當時我就提出質疑了,同事拍着胸脯和我說,有個銀行的請求參數必須得這麼寫,否則就驗證不了。json

我當時還在想,這是 JDK 出的什麼新類型麼,以爲仍是不科學,考慮去跟下同事寫的代碼,而後用他所說的狀況我親自去驗證一下。後端

看了下,這是個老業務系統,同事用了 json-lib 這個包,歷史的江湖確實有這個包的存在,棧長以前也用過,不事後來這玩意就沒怎麼用了,如今都是 GsonJackson 的天下了。微信

以下面 json-lib 例子所示:多線程

public static void main(String[] args) {
    String jsonString = "{\"name\": \"hi\",\"sex\": \"boy\", \"age\": null}";

    JSONObject jsonObject = net.sf.json.JSONObject.fromObject(jsonString);
    Object age = jsonObject.get("age");
    
    // 輸出:null
    System.out.println("age: " + age);

    // 輸出:false
    System.out.println("age == null: " + (age == null));
    
    // 輸出:true
    System.out.println("age.equals(null): " + (age.equals(null)));
}

我天!你們看到結果了吧,問題確實也如同事所說,必定要用 object.equals(null) 寫法才行,不相信結果的你們也能夠親自驗證一下。架構

納了悶了,這樣寫,我傳一個 null 值過去不是報空指針了麼?這樣寫確定有問題,繼續深挖!工具

問題分析

fromObject 方法加載 JSON 串開始源碼深刻分析,找到了這個神奇解析 null 值的源碼:性能

原來,JSON 串中的 null 值被解析成了它內部的 JSONNull 對象,而後再看下這個 JSONNull 的 equals 方法源碼:測試

public boolean equals(Object object) {
    return object == null || 
           object == this || 
           object == instance || 
           object instanceof JSONObject &&
           ((JSONObject)object).isNullObject() ||
           "null".equals(object);
}

問題就出在他所用的 JSON 工具類了!!!

equals 方法被重寫了……終於揭開了 object.equals(null) 的神祕面紗……

再來看下是否有新的更新包:

最新的版本停留在 2010 年 12 月,已是被淘汰的東西了。

另外,json-lib 在 JDK 1.7+ 有性能影響。
推薦閱讀:請不要在 JDK 7+ 中使用這個 JSON 包了

解決方案

方法1:

換掉 object.equals(null),用 JSONNull 的實例去判斷:

public static boolean isNull(Object object){
    return null == object || JSONNull.getInstance().equals(object);
}

方法2:

換掉 json-lib 庫,用主流的 GsonJackson

具體看下這篇:Java經常使用的幾個Json庫,性能強勢對比,另外 FastJson 也不建議用了,漏洞比較多。

這個因爲是老系統,太多業務使用了這個庫,換掉的開發、測試成本和風險比較大,暫時考慮先用方案1先解決這個問題。

關注Java技術棧微信公衆號,棧長將繼續分享好玩的 Java 技術,公衆號第一時間推送,在公衆號後臺回覆:Java,能夠獲取歷史 Java 教程,都是乾貨。

推薦去個人博客閱讀更多:

1.Java JVM、集合、多線程、新特性系列教程

2.Spring MVC、Spring Boot、Spring Cloud 系列教程

3.Maven、Git、Eclipse、Intellij IDEA 系列工具教程

4.Java、後端、架構、阿里巴巴等大廠最新面試題

以爲不錯,別忘了點贊+轉發哦!

相關文章
相關標籤/搜索