Gson解析空字符串异常的处理

更多前沿课程:元宇宙:人类社会数字化转型(开学第一课)--机器学习视频教程-人工智能-优快云程序员研修院

https://edu.youkuaiyun.com/course/detail/30449

SpringBoot2集成Quartz+Vue动态定时任务(前后分离)-quartz vue,springboot 动态定时任务,springboot2 动态定时任务-Java视频教程-后端开发-优快云程序员研修院能独立完整体系化的Quartz真实项目 能使用cron表达式7子表达式进行动态任务处理 能独立完成springboot+quartz的整合,新增,运行,暂停,恢复,暂停,更新工作 能掌握快速高效的学习方法和学习技巧-quartz vue,springboot 动态定时任务,springboot2 动态定时任务正在上传…重新上传取消https://edu.youkuaiyun.com/course/detail/32104icon-default.png?t=M1L8https://edu.youkuaiyun.com/course/detail/32104 SpringCloud+Oauth2+Vue+ElementUI前后端分离快速上手项目实战开发--Java视频教程-后端开发-优快云程序员研修院一个学完能直接通用的微服务项目 一个能直接快速上手,节约学习成本和时间的实战项目 一个企业真实的脱敏后的项目 一个微服务组件协同的前后端分离项目实战教程-正在上传…重新上传取消https://edu.youkuaiyun.com/course/detail/29602icon-default.png?t=M1L8https://edu.youkuaiyun.com/course/detail/29602​​​​​​​

   

 ETL之PDI/Kettle9.x/8.x案例培训实战从入门到应用案例-Array-ETL视频教程-大数据-优快云程序员研修院企业用的最多的Kettle9.x/8.x/7.x,本教程以最新Kettle 9.x进行教学 ETL推送技术开源免费软件排行榜首位的Kettle工具讲解 能独立完成Kettle数据推送项目 能独立进行数据清洗,数据推送,数据迁移工作-Array正在上传…重新上传取消https://edu.youkuaiyun.com/course/detail/31647icon-default.png?t=M1L8https://edu.youkuaiyun.com/course/detail/31647
SpringCloud+Oauth2+Vue+ElementUI前后端分离快速上手项目实战开发--Java视频教程-后端开发-优快云程序员研修院一个学完能直接通用的微服务项目 一个能直接快速上手,节约学习成本和时间的实战项目 一个企业真实的脱敏后的项目 一个微服务组件协同的前后端分离项目实战教程-正在上传…重新上传取消https://edu.youkuaiyun.com/course/detail/29602icon-default.png?t=M1L8https://edu.youkuaiyun.com/course/detail/29602

面对一些不规范的json,我们的gson解析经常会抛出各种异常导致app崩溃,这里可以采取一些措施来避免

关于数组类型的字段解析异常,我尝试了一些方案,但最后都存在问题,如果大家有好的解决方案,希望能贴在下面.不甚感激.
异常示例=>正常json:

{
    "code":0,
    "msg":"ok",
    "data":[    //约定为数组
      {
        "id":5638,
        "newsId":5638
      }
      ]
}

异常json:

{
    "code":0,
    "msg":"ok",
    "data":{}    //返回为对象或者空字符串
}

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字段为空的情况崩溃了



作者:uncochen
链接:https://www.jianshu.com/p/e03a96f5a321
来源:简书
简书著作权归作者所有,任何形式的转载都请联系作者获得授权并注明出处。 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值