前言
在實際開發項目中,服務器經常會用空字符串“” 作為返回結果表示空值,但這在Gson當中就會遇到問題,如果這項數據的類型不是字符串,Gson解析就會報錯
Json異常情況
先來看一個後台返回的json
正常情況下json:
{ "code":0, "msg":"ok", "data":{ "id":5638, "newsId":5638 }}data部分對應的實體類:
public class JsonBean { private int id; private int newsId; public int getId() { return id; } public void setId(int id) { this.id = id; } public int getNewsId() { return newsId; } public void setNewsId(int newsId) { this.newsId = newsId; }}異常情況json (後台數據庫newsId字段未查詢到對應數據):
{ "code":0, "msg":"ok", "data":{ "id":5638, "newsId":"" }}這樣Gson在解析時就會拋出解析錯誤的異常,app崩潰,原因是無法將""轉化為int
json異常的處理
我們期望在後台返回的json異常時,也能解析成功,空值對應的轉換為默認值,如: newsId=0;
這裡排除掉後台開發人員輸出時給你做矯正,還是得靠自己啊---
我們寫一個針對int值的類型轉換器,需要實現Gson的JsonSerializer<T>接口和JsonDeserializer<T> ,即序列化和反序列化接口
public class IntegerDefault0Adapter implements JsonSerializer<Integer>, JsonDeserializer<Integer> { @Override public Integer deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException { try { if (json.getAsString().equals("") || json.getAsString().equals("null")) {//定義為int類型,如果後台返回""或者null,則返回0 return 0; } } catch (Exception ignore) { } try { return json.getAsInt(); } catch (NumberFormatException e) { throw new JsonSyntaxException(e); } } @Override public JsonElement serialize(Integer src, Type typeOfSrc, JsonSerializationContext context) { return new JsonPrimitive(src); }}同理Long及Double類型
double=>
public class DoubleDefault0Adapter implements JsonSerializer<Double>, JsonDeserializer<Double> { @Override public Double deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException { try { if (json.getAsString().equals("") || json.getAsString().equals("null")) {//定義為double類型,如果後台返回""或者null,則返回0.00 return 0.00; } } catch (Exception ignore) { } try { return json.getAsDouble(); } catch (NumberFormatException e) { throw new JsonSyntaxException(e); } } @Override public JsonElement serialize(Double src, Type typeOfSrc, JsonSerializationContext context) { return new JsonPrimitive(src); }}long=>
public class LongDefault0Adapter implements JsonSerializer<Long>, JsonDeserializer<Long> { @Override public Long deserialize(JsonElement json, Type typeOfT, JsonDeserializationContext context) throws JsonParseException { try { if (json.getAsString().equals("") || json.getAsString().equals("null")) {//定義為long類型,如果後台返回""或者null,則返回0 return 0l; } } catch (Exception ignore) { } try { return json.getAsLong(); } catch (NumberFormatException e) { throw new JsonSyntaxException(e); } } @Override public JsonElement serialize(Long src, Type typeOfSrc, JsonSerializationContext context) { return new JsonPrimitive(src); }}所以使用是這樣的:
return new Retrofit.Builder() .client(okHttpClient)//設置網絡訪問框架.addConverterFactory(GsonConverterFactory.create(buildGson()))//添加json轉換框架.addCallAdapterFactory(RxJavaCallAdapterFactory.create())//讓Retrofit支持RxJava .baseUrl(baseUrl) .build();/** * 增加後台返回""和"null"的處理* 1.int=>0 * 2.double=>0.00 * 3.long=>0L * * @return */public static Gson buildGson() { if (gson == null) { gson = new GsonBuilder() .registerTypeAdapter(Integer.class, new IntegerDefault0Adapter()) .registerTypeAdapter(int.class, new IntegerDefault0Adapter()) .registerTypeAdapter(Double.class, new DoubleDefault0Adapter()) .registerTypeAdapter(double.class, new DoubleDefault0Adapter()) .registerTypeAdapter(Long.class, new LongDefault0Adapter()) .registerTypeAdapter(long.class, new LongDefault0Adapter()) .create(); } return gson;}再也不會因為後台json字段為空的情況崩潰了
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能有所幫助,如果有疑問大家可以留言交流。