Moshi三宗罪

Android一直使用的是Goolge的Gson,开始使用kotlin后发现Gson处理kotlin的data class会出现各种空指针,kotlin的非空校验失效;于是开始研究Moshi并使用它。 下面我会简单介绍下moshi的使用,并详细说明从Gson转到Moshi存在的我满意的地方。

使用介绍

Json解析有两种方式,一种是给Bean加注解,一种是使用kotlin的反射,为了简单我这里使用的是反射

  • 添加依赖
kotlin 复制代码
implementation("com.squareup.moshi:moshi-kotlin:1.15.2")
  • 使用
kotlin 复制代码
var moshi =
    Moshi.Builder()
        //Date-long转换
        .add(DateJsonAdapter())
        //moshi官方对kotlin类的处理
        .addLast(KotlinJsonAdapterFactory())
        .build()

fun <T> fromJson(json: String?, type: Type): T? {
    if (json.isNullOrBlank()) return null
    return moshi.adapter<T>(type).fromJson(json)
}

本人不满意的点

一、强制非空判断

json为null但字段非null时,报异常,但我希望的是如果为null则给个默认值,下面的例子中默认指定了val age: Int = 3,而json里是"age":null,解析结果是异常Expected an int but was NULL at path $.age

kotlin 复制代码
private fun testDataClass() {
    val json = """{"name":"wzw","age":null, "gender":1, "son":{}}"""
    //com.squareup.moshi.JsonDataException: Expected an int but was NULL at path $.age
    val fromBeanJson = MoshiUtil.fromBeanJson<DataWen>(json)
    println("moshi: $fromBeanJson")
}
data class DataWen(
    val name: String? = null,
    //如果为null则使用默认值,但moshi做不到
    val age: Int = 3,
    val gender: Int? = null
)

二、无法指定默认值

json为null但字段可null且有默认值,默认值失效,下面的例子中val age: Int? = 3 age可空且指定里默认值,但由于json里的age=null所以最终解析出来的age=null;这个问题和上面第一个问题类似,我希望的是null的时候能使用默认值,但是moshi做不到。

kotlin 复制代码
private fun testDataClass() {
    val json = """{"name":"wzw","age":null, "gender":1, "son":{}}"""
    val fromBeanJson = MoshiUtil.fromBeanJson<DataWen>(json)
    //DataWen(name=wzw, age=null, gender=1)
    println("moshi: $fromBeanJson")
}
data class DataWen(
    val name: String? = null,
    val age: Int? = 3,
    val gender: Int? = null
)

为什么我一定要纠结这个默认值?

因为:

  1. Gson支持默认值,当为null时可以使用默认值,达到对象不为null的效果;
  2. App无法保证后台接口返回的json一定是自己想要的,必须要有容错否则崩溃了就是移动端开发人员的锅;而Moshi的容错能力太差了

三、后台没返回该字段时报错

json没有该字段,但bean中定义了该字段非空,则报错Required value 'age' missing at $。 如下面的例子中,json里没有age字段,但DataWen里定义了非null的age:val age: Int,Moshi在解析时就会报错。

当然,要解决这个问题很简单,只要age给个默认值就可以,如val age: Int = 18。把这个问题列出来是想说Moshi有这个问题,使用时要注意!

kotlin 复制代码
private fun 测试dataClass() {
    val json = """{"name":"wzw", "gender":1, "son":{}}"""
    //com.squareup.moshi.JsonDataException: Required value 'age' missing at $
    val fromBeanJson = MoshiUtil.fromBeanJson<DataWen>(json)
    println("moshi: $fromBeanJson")
}
data class DataWen(
    val name: String? = null,
    val age: Int,
    val gender: Int? = null
)

四、繁琐或耗时等

Moshi的解析方式有两种,一种是通过注解,需要在每个类上加注解,听说解析速度比Gson快;另一种是通过反射,速度比Gson慢,尤其是第一次解析,甚至比Gson慢十几倍,而且反射需要引入kotlin的反射库,jar包2.5M左右;

  • 注解解析需要
kotlin 复制代码
//需要加注解
@JsonClass(generateAdapter = true)
data class DataWen(
    val name: String? = null,
    val age: Int = 3,
    val gender: Int? = null
)
  • 反射解析初次速度慢

这里一直强调初次/第一次是因为kotlin反射sdk第一次使用有一个初始化过程,这个过程耗时,但是即使第二次第三次使用,moshi的解析也不会比gson快,因为kotlin反射本来就比java的慢;---不只是理解的对不对😄

ini 复制代码
//耗时
gson: DataWen(name=wzw, age=null, gender=1) take time: 23 
moshi: DataWen(name=wzw, age=null, gender=1) take time: 301

总结

  • Moshi的容错能力不行,App如果要使用moshi,必须要注意上面这些问题,也可以自己写Adapter规避上面这些问题,但这个会影响性能;

  • 如果决定继续使用Gson,则不能使用kotlin的数据类data class,要改为普通的class,且要有默认空构造,且字段不能是val

kotlin 复制代码
data class DataWen(
    val name: String? = null,
    val age: Int,
    val gender: Int? = null
)
//改为
class DataWen(
    var name: String? = null,
    var age: Int,
    var gender: Int? = null
)
  • 以上列出的所有问题通过自定义Adapter都可以解决,就看你愿不愿意折腾;

要解决可参考:github上的moshi-nullsafe

相关推荐
-SOLO-4 小时前
备份apk 工具
android
私人珍藏库8 小时前
【Android】BotHub-多模型AI机器人聚合库-内置免费模型
android·人工智能·智能手机·app·工具·多功能
普马萨特9 小时前
Wi-Fi 扫描频率限制与 Android 演进全解析
android
张拭心9 小时前
Android 17 新特性:后台音频交互限制加强
android·前端
张拭心10 小时前
Android 17 新特性:ProfilingManager 新触发器
android·前端
张拭心10 小时前
Android 17 新特性:MessageQueue 无锁实现
android·前端
brycegao10 小时前
如何搭建标准化 Git 工具流,保障 Android 团队代码质量
android·ci/cd
AI科技星10 小时前
数术江湖·全卷合集 - 硬核江湖・数理史诗
android·人工智能·架构·概率论·学习方法
五月君_10 小时前
安卓也支持了!微信链接 Claude Code 保姆级教程
android·微信
柚鸥ASO优化10 小时前
一篇讲透安卓ASO!开发者千万别只盯着iOS了
android·ios·aso优化