本篇內(nèi)容主要講解“為什么不使用fastjson”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學(xué)習(xí)“為什么不使用fastjson”吧!
成都創(chuàng)新互聯(lián)提供高防服務(wù)器、云服務(wù)器、香港服務(wù)器、四川電信科技城機房等
究其原因,是fastjson漏洞頻發(fā),導(dǎo)致了公司內(nèi)部需要頻繁的督促各業(yè)務(wù)線升級fastjson版本,來防止安全問題。
fastjson在2020年頻繁暴露安全漏洞,此漏洞可以繞過autoType開關(guān)來實現(xiàn)反序列化遠程代碼執(zhí)行并獲取服務(wù)器訪問權(quán)限。
從2019年7月份發(fā)布的v1.2.59一直到2020年6月份發(fā)布的 v1.2.71 ,每個版本的升級中都有關(guān)于AutoType的升級,涉及13個正式版本。
fastjson中與AutoType相關(guān)的版本歷史:
1.2.59發(fā)布,增強AutoType打開時的安全性 fastjson
1.2.60發(fā)布,增加了AutoType黑名單,修復(fù)拒絕服務(wù)安全問題 fastjson
1.2.61發(fā)布,增加AutoType安全黑名單 fastjson
1.2.62發(fā)布,增加AutoType黑名單、增強日期反序列化和JSONPath fastjson
1.2.66發(fā)布,Bug修復(fù)安全加固,并且做安全加固,補充了AutoType黑名單 fastjson
1.2.67發(fā)布,Bug修復(fù)安全加固,補充了AutoType黑名單 fastjson
1.2.68發(fā)布,支持GEOJSON,補充了AutoType黑名單
1.2.69發(fā)布,修復(fù)新發(fā)現(xiàn)高危AutoType開關(guān)繞過安全漏洞,補充了AutoType黑名單
1.2.70發(fā)布,提升兼容性,補充了AutoType黑名單
1.2.71發(fā)布,補充安全黑名單,無新增利用,預(yù)防性補充
相比之下,其他的json框架,如Gson和Jackson,漏洞數(shù)量少很多,高危漏洞也比較少,這是公司想要替換框架的主要原因。
本文主要討論Gson替換fastjson框架的實戰(zhàn)問題,所以在這里不展開詳細討論各種json框架的優(yōu)劣,只給出結(jié)論。
經(jīng)過評估,主要有Jackson和Gson兩種json框架放入考慮范圍內(nèi),與fastjson進行對比。
速度快
fastjson相對其他JSON庫的特點是快,從2011年fastjson發(fā)布1.1.x版本之后,其性能從未被其他Java實現(xiàn)的JSON庫超越。
使用廣泛
fastjson在阿里巴巴大規(guī)模使用,在數(shù)萬臺服務(wù)器上部署,fastjson在業(yè)界被廣泛接受。在2012年被開源中國評選為最受歡迎的國產(chǎn)開源軟件之一。
測試完備
fastjson有非常多的testcase,在1.2.11版本中,testcase超過3321個。每次發(fā)布都會進行回歸測試,保證質(zhì)量穩(wěn)定。
使用簡單
fastjson的API十分簡潔。
容易使用 - jackson API提供了一個高層次外觀,以簡化常用的用例。
無需創(chuàng)建映射 - API提供了默認的映射大部分對象序列化。
性能高 - 快速,低內(nèi)存占用,適合大型對象圖表或系統(tǒng)。
干凈的JSON - jackson創(chuàng)建一個干凈和緊湊的JSON結(jié)果,這是讓人很容易閱讀。
不依賴 - 庫不需要任何其他的庫,除了JDK。
提供一種機制,使得將Java對象轉(zhuǎn)換為JSON或相反如使用toString()以及構(gòu)造器(工廠方法)一樣簡單。
允許預(yù)先存在的不可變的對象轉(zhuǎn)換為JSON或與之相反。
允許自定義對象的表現(xiàn)形式
支持任意復(fù)雜的對象
輸出輕量易讀的JSON
同事撰寫的性能對比源碼:
https://github.com/zysrxx/json-comparison
本文不詳細討論性能的差異,畢竟這其中涉及了很多各個框架的實現(xiàn)思路和優(yōu)化,所以只給出結(jié)論:
1.序列化單對象性能Fastjson > Jackson > Gson,其中Fastjson和Jackson性能差距很小,Gson性能較差
2.序列化大對象性能Jackson> Fastjson > Gson ,序列化大Json對象時Jackson> Gson > Fastjson,Jackson序列化大數(shù)據(jù)時性能優(yōu)勢明顯
3.反序列化單對象性能 Fastjson > Jackson > Gson , 性能差距較小
4.反序列化大對象性能 Fastjson > Jackson > Gson , 性能差距較很小
企業(yè)項目或者說大型項目的特點:
所以對于大型項目,想要做到將底層的fastjson遷移到gson是一件復(fù)雜且痛苦的事情,其實對于其他依賴的替換,也都一樣。
我總結(jié)了如下幾個在替換項目依賴過程中要特別重視的問題。
再怎么謹慎都不為過,如果你要更改的項目是非常重要的業(yè)務(wù),那么一旦犯下錯誤,代價是非常大的。并且,對于業(yè)務(wù)方和產(chǎn)品團隊來說,沒有新的功能上線,但是系統(tǒng)卻炸了,是一件“無法忍受”的事情。盡管你可能覺得很委屈,因為只有你或者你的團隊知道,雖然業(yè)務(wù)看上去沒變化,但是代碼底層已經(jīng)發(fā)生了翻天覆地的變化。
所以,謹慎點!
在依賴替換的過程中,需要做好項目的規(guī)劃,比如分模塊替換,嚴格細分排期。
把前期規(guī)劃做好,開發(fā)和測試才能有條不紊的進行工作。
開發(fā)之間,需要提前溝通好開發(fā)注意事項,比如依賴版本問題,防止由多個開發(fā)同時修改代碼,最后發(fā)現(xiàn)使用的版本不同,接口用法都不同這種很尷尬,并且要花額外時間處理的事情。
而對于測試,更要事先溝通好。一般來說,測試不會太在意這種對于業(yè)務(wù)沒有變化的技術(shù)項目,因為既不是優(yōu)化速度,也不是新功能。但其實遷移涉及到了底層,很容易就出現(xiàn)BUG。要讓測試團隊了解更換項目依賴,是需要大量的測試時間投入的,成本不亞于新功能,讓他們盡量重視起來。
上面說到測試團隊需要投入大量工時,這些工時主要都用在項目功能的整體回歸上,也就是回歸測試。
當然,不只是業(yè)務(wù)回歸測試,如果有條件的話,要做接口回歸測試。
如果公司有接口管理平臺,那么可以極大提高這種項目測試的效率。
打個比方,在一個模塊修改完成后,在測試環(huán)境(或者沙箱環(huán)境),部署一個線上版本,部署一個修改后的版本,直接將接口返回數(shù)據(jù)進行對比。一般來說是Json對比,網(wǎng)上也有很多的Json對比工具:
https://www.sojson.com/
正如上面描述的Gson和Fastjson性能對比,替換框架需要注意框架之間的性能差異,尤其是對于流量業(yè)務(wù),也就是高并發(fā)項目,響應(yīng)時間如果發(fā)生很大的變化會引起上下游的注意,導(dǎo)致一些額外的后果。
這里總結(jié)了兩種json框架常用的方法,貼出詳細的代碼示例,幫助大家快速的上手Gson,無縫切換!
String jsonCase = "[{\"id\":10001,\"date\":1609316794600,\"name\":\"小明\"},{\"id\":10002,\"date\":1609316794600,\"name\":\"小李\"}]";
// fastjson
JSONArray jsonArray = JSON.parseArray(jsonCase);
System.out.println(jsonArray);
System.out.println(jsonArray.getJSONObject(0).getString("name"));
System.out.println(jsonArray.getJSONObject(1).getString("name"));
// 輸出:
// [{"date":1609316794600,"name":"小明","id":10001},{"date":1609316794600,"name":"小李","id":10002}]
// 小明
// 小李
// Gson
JsonArray jsonArrayGson = gson.fromJson(jsonCase, JsonArray.class);
System.out.println(jsonArrayGson);
System.out.println(jsonArrayGson.get(0).getAsJsonObject().get("name").getAsString());
System.out.println(jsonArrayGson.get(1).getAsJsonObject().get("name").getAsString());
// 輸出:
// [{"id":10001,"date":1609316794600,"name":"小明"},{"id":10002,"date":1609316794600,"name":"小李"}]
// 小明
// 小李
看得出,兩者區(qū)別主要在get各種類型上,Gson調(diào)用方法有所改變,但是變化不大。
那么,來看下空對象反序列化會不會出現(xiàn)異常:
String jsonObjectEmptyCase = "{}";
// fastjson
JSONObject jsonObjectEmpty = JSON.parseObject(jsonObjectEmptyCase);
System.out.println(jsonObjectEmpty);
System.out.println(jsonObjectEmpty.size());
// 輸出:
// {}
// 0
// Gson
JsonObject jsonObjectGsonEmpty = gson.fromJson(jsonObjectEmptyCase, JsonObject.class);
System.out.println(jsonObjectGsonEmpty);
System.out.println(jsonObjectGsonEmpty.size());
// 輸出:
// {}
// 0
沒有異常,開心。
看看空數(shù)組呢,畢竟[]感覺比{}更加容易出錯。
String jsonArrayEmptyCase = "[]";
// fastjson
JSONArray jsonArrayEmpty = JSON.parseArray(jsonArrayEmptyCase);
System.out.println(jsonArrayEmpty);
System.out.println(jsonArrayEmpty.size());
// 輸出:
// []
// 0
// Gson
JsonArray jsonArrayGsonEmpty = gson.fromJson(jsonArrayEmptyCase, JsonArray.class);
System.out.println(jsonArrayGsonEmpty);
System.out.println(jsonArrayGsonEmpty.size());
// 輸出:
// []
// 0
兩個框架也都沒有問題,完美解析。
解析泛型是一個非常常用的功能,我們項目中大部分fastjson代碼就是在解析json和Java Bean。
// 實體類
User user = new User();
user.setId(1L);
user.setUserName("馬云");
// fastjson
List userListResultFastjson = JSONArray.parseArray(JSON.toJSONString(userList), User.class);
List userListResultFastjson2 = JSON.parseObject(JSON.toJSONString(userList), new TypeReference>(){});
System.out.println(userListResultFastjson);
System.out.println("userListResultFastjson2" + userListResultFastjson2);
// 輸出:
// userListResultFastjson[User [Hash = 483422889, id=1, userName=馬云], null]
// userListResultFastjson2[User [Hash = 488970385, id=1, userName=馬云], null]
// Gson
List userListResultTrue = gson.fromJson(gson.toJson(userList), new TypeToken>(){}.getType());
System.out.println("userListResultGson" + userListResultGson);
// 輸出:
// userListResultGson[User [Hash = 1435804085, id=1, userName=馬云], null]
可以看出,Gson也能支持泛型。
這一點fastjson和Gson有區(qū)別,Gson不支持直接將List寫入value,而fastjson支持。
所以Gson只能將List解析后,寫入value中,詳見如下代碼:
// 實體類
User user = new User();
user.setId(1L);
user.setUserName("馬云");
// fastjson
JSONObject jsonObject1 = new JSONObject();
jsonObject1.put("user", user);
jsonObject1.put("userList", userList);
System.out.println(jsonObject1);
// 輸出:
// {"userList":[{"id":1,"userName":"馬云"},null],"user":{"id":1,"userName":"馬云"}}
// Gson
JsonObject jsonObject = new JsonObject();
jsonObject.add("user", gson.toJsonTree(user));
System.out.println(jsonObject);
// 輸出:
// {"user":{"id":1,"userName":"馬云"},"userList":[{"id":1,"userName":"馬云"},null]}
如此一來,Gson看起來就沒有fastjson方便,因為放入List是以gson.toJsonTree(user)
的形式放入的。這樣就不能先入對象,在后面修改該對象了。(有些同學(xué)比較習(xí)慣先放入對象,再修改對象,這樣的代碼就得改動)
駝峰轉(zhuǎn)換下劃線依靠的是修改Gson的序列化模式,修改為LOWER_CASE_WITH_UNDERSCORES
GsonBuilder gsonBuilder = new GsonBuilder();
gsonBuilder.setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES);
Gson gsonUnderScore = gsonBuilder.create();
System.out.println(gsonUnderScore.toJson(user));
// 輸出:
// {"id":1,"user_name":"馬云"}
下面整理了我們在公司項目遷移Gson過程中,踩過的坑,這些坑現(xiàn)在寫起來感覺沒什么技術(shù)含量。但是這才是我寫這篇文章的初衷,幫助大家把這些很難發(fā)現(xiàn)的坑避開。
這些問題有的是在測試進行回歸測試的時候發(fā)現(xiàn)的,有的是在自測的時候發(fā)現(xiàn)的,有的是在上線后發(fā)現(xiàn)的,比如Swagger掛了這種不會去測到的問題。
不知道大家想過一個問題沒有,如果你的項目里有緩存系統(tǒng),使用fastjson寫入的緩存,在你切換Gson后,需要用Gson解析出來。所以就一定要保證兩個框架解析邏輯是相同的,但是,顯然這個愿望是美好的。
在測試過程中,發(fā)現(xiàn)了Date類型,在兩個框架里解析是不同的方式。
導(dǎo)致了Gson在反序列化這個json的時候,直接報錯,無法轉(zhuǎn)換為Date。
解決方案:
新建一個專門用于解析Date類型的類:
import com.google.gson.TypeAdapter;
import com.google.gson.stream.JsonReader;
import com.google.gson.stream.JsonWriter;
import java.io.IOException;
import java.util.Date;
public class MyDateTypeAdapter extends TypeAdapter {
@Override
public void write(JsonWriter out, Date value) throws IOException {
if (value == null) {
out.nullValue();
} else {
out.value(value.getTime());
}
}
@Override
public Date read(JsonReader in) throws IOException {
if (in != null) {
return new Date(in.nextLong());
} else {
return null;
}
}
}
接著,在創(chuàng)建Gson時,把他放入作為Date的專用處理類:
Gson gson = new GsonBuilder().registerTypeAdapter(Date.class,new MyDateTypeAdapter()).create();
這樣就可以讓Gson將Date處理為Unix。
當然,這只是為了兼容老的緩存,如果你覺得你的倉庫沒有這方面的顧慮,可以忽略這個問題。
切換到Gson后,使用SpringBoot搭建的Web項目的接口直接請求不了了。報錯類似:
org.springframework.http.converter.HttpMessageNotWritableException
因為SpringBoot默認的Mapper是Jackson解析,我們切換為了Gson作為返回對象后,Jackson解析不了了。
解決方案:
application.properties里面添加:
#Preferred JSON mapper to use for HTTP message conversion
spring.mvc.converters.preferred-json-mapper=gson
這個問題和上面的SpringBoot異常類似,是因為在SpringBoot中引入了Gson,導(dǎo)致 swagger 無法解析 json。
采用類似下文的解決方案(添加Gson適配器):
http://yuyublog.top/2018/09/03/springboot%E5%BC%95%E5%85%A5swagger/
@Configuration
public class GsonSwaggerConfig {
//設(shè)置swagger支持gson
@Bean
public IGsonHttpMessageConverter IGsonHttpMessageConverter() {
return new IGsonHttpMessageConverter();
}
}
public class IGsonHttpMessageConverter extends GsonHttpMessageConverter {
public IGsonHttpMessageConverter() {
//自定義Gson適配器
super.setGson(new GsonBuilder()
.registerTypeAdapter(Json.class, new SpringfoxJsonToGsonAdapter())
.serializeNulls()//空值也參與序列化
.create());
}
}
public class SpringfoxJsonToGsonAdapter implements JsonSerializer {
@Override
public JsonElement serialize(Json json, Type type, JsonSerializationContext jsonSerializationContext) {
return new JsonParser().parse(json.value());
}
}
有時候,我們會在入?yún)⑹褂妙愃疲?/p>
public ResponseResult submitAudit(@RequestBody JsonObject jsonObject) {}
如果使用這種代碼,其實就是使用Gson來解析json字符串。但是這種寫法的風(fēng)險是很高的,平常請大家盡量避免使用JsonObject直接接受參數(shù)。
在Gson中,JsonObject若是有數(shù)字字段,會統(tǒng)一序列化為double,也就是會把count = 0
這種序列化成count = 0.0
。
為何會有這種情況?簡單的來說就是Gson在將json解析為Object類型時,會默認將數(shù)字類型使用double轉(zhuǎn)換。
如果Json對應(yīng)的是Object類型,最終會解析為Map
類型;其中Object類型跟Json中具體的值有關(guān),比如雙引號的""值翻譯為STRING。我們可以看下數(shù)值類型(NUMBER)全部轉(zhuǎn)換為了Double類型,所以就有了我們之前的問題,整型數(shù)據(jù)被翻譯為了Double類型,比如30變?yōu)榱?0.0。
可以看下Gson的ObjectTypeAdaptor類,它繼承了Gson的TypeAdaptor抽象類:
解決方案:
第一個方案:把入?yún)⒂脤嶓w類接收,不要使用JsonObject
第二個方案:與上面的解決Date類型問題類似,自己定義一個Adaptor,來接受數(shù)字,并且處理。這種想法我覺得可行但是難度較大,可能會影響到別的類型的解析,需要在設(shè)計適配器的時候格外注意。
到此,相信大家對“為什么不使用fastjson”有了更深的了解,不妨來實際操作一番吧!這里是創(chuàng)新互聯(lián)網(wǎng)站,更多相關(guān)內(nèi)容可以進入相關(guān)頻道進行查詢,關(guān)注我們,繼續(xù)學(xué)習(xí)!