四年前老架構師的話沒聽進去,現在出問題了

一、FastJson爲何

FastJSON在官方的github上面是這樣子說的

Fastjson is a Java library that can be used to convert Java Objects into their JSON representation. It can also be used to convert a JSON string to an equivalent Java object. Fastjson can work with arbitrary Java objects including pre-existing objects that you do not have source-code of.

翻譯:Fastjson是一個Java庫,可用於將Java對象轉換爲其JSON表示形式。它還可以用於將JSON字符串轉換爲等效的Java對象。Fastjson可以與任意Java對象一起使用,包括您沒有源代碼的現有對象。

FastJson是Java程序員常用到的類庫之一,相信點開這個頁面的你,也肯定是程序員朋友。正如其名,“快”是其主要賣點。

四年前老架構師的話沒聽,現在出問題了,FastJson,到了說再見了

 

但是,其實在久遠 的4年前的一個項目中,有一位老架構前輩發現我們項目中用FastJSON,跟我們說,FastJSON的源碼比較噁心,而且有很多漏洞,用的時候注意點,當時也不太懂,沒怎麼在意。

直到最近幾個月FastJSON連續爆出兩個服務器級別的漏洞。。。

FastJSON使用體驗

FastJSON目前在使用體驗上,確實很不錯,API簡單,上手快,適應力強。

二、真的很快嗎?

沒有調研就沒有發言權,本着“追求真理”的初心,來一輪簡單的測試。對比對象選擇應用最廣泛的Jackson和Google出品的Gson。測試環境選擇JDK 8,AMD 3700X,3200MHZ內存。簡化實驗,只測試簡單對象和複雜對象的String轉對象、對象轉String,調用1千萬次的對比結果如下(時間單位是毫秒):

四年前老架構師的話沒聽,現在出問題了,FastJson,到了說再見了

 

從測試結果看,FastJson確實是最快的,但僅比Jackson快20%左右,Google的Gson是最慢的,差距較大。讀到這裏,是不是覺得選擇FastJson肯定沒錯啊!如果面試官問爲什麼選擇FastJson?因爲快!這一個理由就可以把他頂回去了。

這裏的調查研究並不是很充分,沒有對內存佔用、大文檔的測試。

在現代應用程序中,即使最慢的Gson,也是滿足需求的;解析文檔速度的快慢,並不能作爲選型的唯一標準,可能連主要標準都算不上。對IO優化,並行處理等優化措施,比選用一個更快的庫更有效。

三、FastJson並沒有那麼流行

然而,FastJson並沒有那麼流行,有一個最直觀的數據,那就是在Maven的中的引用量,和Jackson和Gson不在一個數量級,和Jackson強大的家族更沒法比。

四年前老架構師的話沒聽,現在出問題了,FastJson,到了說再見了

 

難道我用了一個假的流行的國產類庫?在知乎看到了一篇帖子,討論爲什麼外國友人不喜歡FastJson。結論就是FastJson是個代碼質量不高的國產類庫。完全顛覆了我的認知,因爲在我的項目中,是經常使用FastJson的,並沒有出現什麼Bug,而且這段評論是在2016年寫的。

四年前老架構師的話沒聽,現在出問題了,FastJson,到了說再見了

 

抱着懷疑的態度,打開FastJson的地址,看到大家提的Issues。竟然有1283個未解決的Issues。紅框標識出來的,我自己拿去研究下,因爲我看到下面還有人提了一樣的問題。

四年前老架構師的話沒聽,現在出問題了,FastJson,到了說再見了

 

測試代碼如下:

 try {
      String time = "1970-01-01 00:00:00";
      com.alibaba.fastjson.JSONObject jsonObject = new com.alibaba.fastjson.JSONObject();
      jsonObject.put("time", time);

      Timestamp timestamp = jsonObject.getTimestamp("time");
      System.out.println("time:" + timestamp);
  } catch (Exception e) {
      e.printStackTrace();
  }

果然,在採用了最新版本的類庫後,如問題描述的,還是有異常。於是就看到了如下的源代碼:

if (strVal.endsWith(".000000000")) {
    strVal = strVal.substring(0, strVal.length() - 10);
} else if (strVal.endsWith(".000000")) {
    strVal = strVal.substring(0, strVal.length() - 7);
}

if (strVal.length() == 29 && strVal.charAt(4) == '-' && strVal.charAt(7) == '-' && strVal.charAt(10) == ' ' && strVal.charAt(13) == ':' && strVal.charAt(16) == ':' && strVal.charAt(19) == '.') {
    int year = num(strVal.charAt(0), strVal.charAt(1), strVal.charAt(2), strVal.charAt(3));
    int month = num(strVal.charAt(5), strVal.charAt(6));
    int day = num(strVal.charAt(8), strVal.charAt(9));
    int hour = num(strVal.charAt(11), strVal.charAt(12));
    int minute = num(strVal.charAt(14), strVal.charAt(15));
    int second = num(strVal.charAt(17), strVal.charAt(18));
    int nanos = num(strVal.charAt(20), strVal.charAt(21), strVal.charAt(22), strVal.charAt(23), strVal.charAt(24), strVal.charAt(25), strVal.charAt(26), strVal.charAt(27), strVal.charAt(28));
  return new Timestamp(year - 1900, month - 1, day, hour, minute, second, nanos);
}

這段代碼有嚴重的邏輯錯誤,這樣錯誤的格式,例如:

“1970-01-01 00:00:00.000000000.000000000”

或者

“1970-01-01 00:00:00.000000000.000000”

也能轉換成功,而一些正確的格式,例如:

“1970-01-01 00:00:00”,“1970-01-01 00:00:00.000”

卻轉換失敗。

結合知乎上網友的點評,我本人也覺得FastJson並沒有那麼優秀,另一些深入的點評,例如ASM,我的理解並不深,就不做測試了。

四、棄坑fastjson

在我負責的項目中,因爲SpringBoot相關的框架中,應用了Jackson,本着“最少依賴”的原則,json解析應用了Jackson。但是很多同事的代碼中,也用了Gson和Fastjson,當然,是沒有嚴格規範要求的結果。

通過今天的一個小小研究,Jackson的流行,是有着內在的原因的。在我們以後的項目中,主推Jackson,逐漸的淘汰Fastjson。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章