搜狐新闻客户端使用Kotlin之后对JSON解析框架的探索

04ff17e28a31743c7cbf369a208e04b6.jpeg

8807d0b88834fa4655f5b6e6d1681458.gif

本文字数:7488

预计阅读时间:45分钟

39d3db16eedf2142afb870aa5fd11432.png

01

引言

2017年Google发布Kotlin语言之后,Android开发由原来的Java开始向Kotlin
过度,目前绝大部分Android开发岗位基本要求就是熟练使用Kotlin。事实上,很多有着多年历史的项目一开始是Java开发的,在Kotlin日渐趋于Android开发主流的过程中,混合开发成为许多项目的首选。我们的项目也是采用混合开发,面对拥有沉重历史包袱的代码,想用Kotlin重构却不得不考虑时间成本和人力成本,但又不想放弃Kotlin开发的优势,所以新业务均采用Kotlin开发。

Json就不过多介绍了,大家耳熟能详,相信很多伙伴项目中的Json解析依旧在使用FastJson或者Gson等第三方框架进行数据解析,当我们混合开发之后,你会发现Kotlin的数据类写起来很方便,但是将Json解析为数据类对象时出现的问题会让你很头大,尤其是开启混淆之后,各种各样的问题甚至程序崩溃随之出现,随着程序的崩溃,你的内心渐渐开始崩溃,不禁发出疑问,数据类不好用吗?

02

常见Json解析框架

  • FastJson:阿里巴巴公司所开发的 JSON 库,由Java语言开发,对Kotlin有一定的支持,目前FastJson1不在维护,转而维护FastJson2,称FastJson2是为未来十年打造一款高性能的Json解析框架,核心原理是反射。

  • Jackson:Spring 默认的 JSON 库,GitHub上面介绍其是"the best JSON parser for Java"(最好的Java Json解析器),支持解析多种数据格式,核心原理也是反射

    Gson:Google官方开发维护的 JSON 库,目的是为Java开发提供数据解析支持,功能非常强大,核心原理依旧是反射。

    Kotlinx.serialization:Kotlin官方开发的基于Kotlin的序列化与反序列化库,它包括了用于生成代码的插件、具有核心序列化API的运行时库以及具有各种序列化格式的支持库。编译器插件为可序列化的类生成访问者和序列化器,核心原理利用了Kotlin在编译时能够生成代码的特性,从而避免了反射的使用。

03

发生了什么问题

环境:Android Kotlin FastJson1.1.56

背景:新需求初步测试完成,即将合版,所有的Case测试通过,在最后一天用Release包测试时,发现崩溃,经过好几个人三个小时的加班问题解决了。

原因:当客户端与服务器交换数据时,使用数据类作为数据存储的方式,用FastJson将Json字符串解析为对象,用于业务逻辑的使用。Release包开启了代码混淆,导致FastJson在解析时,无法利用反射反射到数据类的构造器,从而抛出异常导致解析失败。

解决方案:改变混淆规则,数据类不进行混淆,因为数据类val关键字修饰的变量不会生成对应的set方法,而FastJson在创建对象后进行赋值时需要调用set方法为其赋值,因此还得将修饰符改为var,并且需要添加默认值,否则在反射创建对象的过程中会抛出异常,导致程序崩溃。

04

FastJson要升级新版本吗

1、问题场景再现


废话不多说,首先我们看一下问题出现的异常:default constructor not found.

888b6e79ad6fb8e5b7ea7d11fb379489.png其中ComplexEntity是我们的数据类,由val关键字修饰,并且没有任何默认值,代码如下:

import java.io.Serializabledata class ComplexEntity(val id: Int,val name: String,val score: Float,val userInfo: UserInfo
) : Serializabledata class UserInfo(val userName: String,val userAge: Int,val userFriends: List<FriendsInfo>
) : Serializabledata class FriendsInfo(val phoneNumber: String,val userTag: String,val groupId: Int
) : Serializable

我们的用法也很简单,就是要把一个Json字符串解析成为一个ComplexEntity类型的对象,代码如下,但是这里会抛出开头提到的异常。

val parseObject = JSON.parseObject(json, ComplexEntity::class.java)

我们来看看堆栈报错中JavaBeanInfo.build()方法中为什么会导致该错误:

public static JavaBeanInfo build(Class<?> clazz, //int classModifiers, //Type type, //boolean fieldOnly, //boolean jsonTypeSupport, //boolean jsonFieldSupport, //boolean fieldGenericSupport, //PropertyNamingStrategy propertyNamingStrategy
) {
List<FieldInfo> fieldList = new ArrayList<FieldInfo>();// DeserializeBeanInfo beanInfo = null;
Constructor<?> defaultConstructor = null;
if ((classModifiers & Modifier.ABSTRACT) == 0) {try {defaultConstructor = clazz.getDeclaredConstructor();} catch (Exception e) {// skip}if (defaultConstructor == null) {if (clazz.isMemberClass() && (classModifiers & Modifier.STATIC) == 0) { // for inner none static classfor (Constructor<?> constructor : clazz.getDeclaredConstructors()) {Class<?>[] parameterTypes = constructor.getParameterTypes();if (parameterTypes.length == 1 && parameterTypes[0].equals(clazz.getDeclaringClass())) {defaultConstructor = constructor;break;}}}}
}Constructor<?> creatorConstructor = null;
Method[] methods = fieldOnly //? null //: clazz.getMethods();final Field[] declaredFields = clazz.getDeclaredFields();if (defaultConstructor == null // && !(clazz.isInterface() || (classModifiers & Modifier.ABSTRACT) != 0) //) {creatorConstructor = null;for (Constructor<?> constructor : clazz.getDeclaredConstructors()) {JSONCreator annotation = constructor.getAnnotation(JSONCreator.class);if (annotation != null) {if (creatorConstructor != null) {throw new JSONException("multi-json creator");}creatorConstructor = constructor;break;}}if (creatorConstructor != null) {...... 此处省略构造器不为空的时候直接通过构造器创建对象return new JavaBeanInfo(clazz, null, creatorConstructor, null, fields, sortedFields, jsonType);}Method factoryMethod = null;{for (Method method : methods) {if ((!Modifier.isStatic(method.getModifiers())) //|| !clazz.isAssignableFrom(method.getReturnType()) //) {continue;}JSONCreator annotation = method.getAnnotation(JSONCreator.class);if (annotation != null) {if (factoryMethod != null) {throw new JSONException("multi-json creator");}factoryMethod = method;break;}}}if (factoryMethod != null) {......此处省略创建beanInfo的过程return beanInfo;}//抛出该异常原因是factoryMethod 没有被赋值throw new JSONException("default constructor not found. " + clazz);
}

此方法的目的是通过反射获取传入的clazz类的构造器,从而为后面解析创建对象时使用。在Json串解析时将值赋值给对应属性时,也会提前通过反射去构造一个FileInfo,它里面保存了对应的属性名和set方法。在赋值时会调用对应的set方法。

840ddbf9558d54c8e3d385cec67a14e7.png

而我们的数据类声明的是val类型的变量,那么编译器会帮我们生成对应的get、toString方法等,通过Android Studio查看数据类字节码反编译后的结果,可以看到val所修饰的属性不会生成对应的set方法,并且也不会生成默认的无参构造器,只有一个全参构造器,那么执行上面的代码时:构造器获取不到会抛出异常,set方法获取不到无法给属性赋值,因此打断了框架利用反射解析JSON的施法。通过上面的结果我们可以对数据类改造如下:

data class ComplexEntity(var id: Int = 0,var name: String = "",var score: Float = 0f,var userInfo: UserInfo = UserInfo()
) : Serializabledata class UserInfo(var userName: String ="",var userAge: Int = 0,var userFriends: List<FriendsInfo> = mutableListOf()
) : Serializabledata class FriendsInfo(var phoneNumber: String = "",var userTag: String = "",var groupId: Int = 0
) : Serializable

这样写之后,在Json解析为对象时,就能够通过反射正确反射出类的构造器以及set方法,并正确为每个属性进行赋值了。

但是依然有一个问题,当我们的Json字符串中某个String或者对象类型的值为null时,例如:{......,"name":null,......}这样的Json在进行解析时则会出现以下异常:

f74b4042184e1e3cc12b48415c6bb7a9.png

很显然,是在反射进行赋值的时候将null赋值给了不可为空的属性,导致异常发生。所以说在遇到这种字段时,我们需要将对应的属性设置为可空类型的。

另外就是很多项目在进行release打包后,会混淆代码,但是在Json解析时,如果代码属性名方法名亦或是类名混淆之后,反射还能正常工作吗,当然不能,因此还需要配置混淆规则,Json解析的类不进行混淆。

2、FastJson现状


首先我们先来了解以下FastJson的版本,目前项目中使用的是1.1.56版本,GitHub上FastJson1的最后一个版本是1.2.83,于2022年五月发布,至今一年多没有迭代新的版本。

16ce0b4316035eee1958a08f521e8b56.png

而我们使用的版本更是2017年一月发布的版本,时至今日,已经六年多时间过去了,期间也出过FastJson漏洞的事件。六年里,FastJson团队一致在积极维护更新。随着Kotlin语言的发展,FastJson也对Kotlin进行了一定的支持。

FastJson2 于2022年四月发布,依旧是之前的团队进行开发维护,该框架从官方介绍上说是对性能有了进一步的提升,目的是为下一个10年提供一个高性能的JSON库,覆盖的场景挺多,对Android和Kotlin进行了一定的支持。并且groupId发生了改变,因此升级需要替换包名并且验证所有API的兼容性,具体可参考GitHub详细介绍。

b6f32636c6b850507b144973e8327c8d.png

3、FastJson1与FastJson2对数据类的支持情况


经过测试,不管是FastJson1还是FastJson2都没有很好的兼容Kotlin Data Class,我们在使用过程中,发现新旧最新版本在对Kotlin数据类进行支持时,都需要引入Kotlin-reflect包,这是一个Kotlin实现反射的包,包大小超过2M,在当下包体大小也是一个App的指标,增加2M包体大小固然不是很好的解决方式。我们主要测试的结果如下:

首先,测试FastJson1,版本1.2.83,混淆导致出错的解决方案已经在上面提到,因此这里不在测试混淆后的结果。

●     数据类中的属性由val修饰,无默认值,结果如下:

599af14ae97619c7e48169fd952a4863.png

我们找到JavaBeanInfo类的build方法中,可以看到有下面一段代码:

String[] paramNames = null;
if (kotlin && constructors.length > 0) {paramNames = TypeUtils.getKoltinConstructorParameters(clazz);creatorConstructor = TypeUtils.getKotlinConstructor(constructors, paramNames);TypeUtils.setAccessible(creatorConstructor);
} else {......
}
public static String[] getKoltinConstructorParameters(Class clazz) {if (kotlin_kclass_constructor == null && !kotlin_class_klass_error) {try {Class class_kotlin_kclass = Class.forName("kotlin.reflect.jvm.internal.KClassImpl");kotlin_kclass_constructor = class_kotlin_kclass.getConstructor(Class.class);} catch (Throwable e) {kotlin_class_klass_error = true;}}if (kotlin_kclass_constructor == null) {return null;}......

我们可以看到在反射创建构造器时,如果是Kotlin类,那么则使用Kotlin-reflect反射来获取对应的构造方法,因此看到这里,就知道为什么需要引入Kotlin-reflect才能够解析成功了。因此我们引入kotlin反射相关的依赖,则解析成功。开启混淆之后,由于我们的实体类都实现了java.io.Serializable接口,因此我们的配置如下:

-keep class kotlin.reflect.jvm.** {*;}
-keep class * implements java.io.Serializable {*;}

接下来测试FastJson2对数据类的支持,我们在项目中导入com.alibaba.fastjson2:fastjson2:2.0.34.android4的依赖,该版本适配了Android4。接下来依旧针对上述内容进行测试。

数据类使用val 修饰,没有默认值,未开启混淆并且为导入kotlin-reflect的情况下,运行程序发生如下异常:

ed936c217cf616025297f4cddd5a7cc0.png对发生异常的地方进行调试,再次遇到了一个熟悉的异常,发现是Kotlin数据类的空安全类型导致的,因为此处创建对象时会给属性赋默认值,而在Java当中,String或者其它Object类型的默认值为null,将一个null值赋值给Kotlin的no-null类型的属性便会发生异常。

e04c48638720be06bbc063cc3a3d8a70.png

将数据类中String或者其它引用类型声明为可空类型的属性,再次测试得到的结果时返回了一个只有默认值的对象,解析失败。因为其底层原理依旧是反射,因此导入kotlin-reflect反射依赖,解析成功。既然是一个全新的框架,可以不依赖反射吗?

当然是可以的,我们根据其FastJson核心是反射的原理将数据类进行改造,让编译器为数据类生成默认构造器以及set方法,改造如下:其中default字段在JSON串中并不存在对应的Key-Value,但是解析后默认值依然存在,另外开启混淆之后,将实体类的混淆关闭即可。

data class ComplexEntity(var id: Int = 0,var name: String = "",var score: Float = 0f,var userInfo: UserInfo = UserInfo(),val default: Int = 10012,
) : Serializabledata class UserInfo(var userName: String ="",var userAge: Int = 0,var userFriends: List<FriendsInfo> = mutableListOf()
) : Serializabledata class FriendsInfo(var phoneNumber: String = "",var userTag: String = "",var groupId: Int = 0
) : Serializable

05

Gson如何?

Gson是Google团队开发维护的用于Java中JSON解析的框架,其底层原理依旧使用了反射,具体的反射过程与解析这里就不再分析了;数据类亦能够正常解析为对象,但是当Json串中值出现null之后,他不会进行空安全类型检查,会直接赋值为null。

val json ="{\"id\":100,\"name\":null,\"score\":99.9,\"userInfo\":{\"userAge\":24,\"userFriends\":null,\"userName\":\"Tom\"}}"val parseObject = gson.fromJson(json, ComplexEntity::class.java)
Log.d(TAG, parseObject.toString())
//ComplexEntity(id=100, name=null, score=99.9, userInfo=UserInfo(userName=Tom, userAge=24, userFriends=null))

这样在使用数据类对应的属性时容易导致异常造成程序崩溃。

另外一个就是不支持数据类的默认值,当我们数据类定义有具有默认值的属性,而Json串中没有时,此时我们解析之后会发现我们定义的值变成了该类型的默认值,在某些特殊情况下可能会导致异常。

data class ComplexEntity(val id: Int,val name: String,val score: Float,val userInfo: UserInfo,val default: Int = 1223//默认值为1223  解析后:default属性的值为:0
) : Serializable

06

Kotlinx-Serialization

Kotlinx-Serialization是Kotlin官方序列化库,相比于FastJson,Kotlin-Serializable未使用反射,并且许多生成代码相关的功能添加到了编译器中,GitHub地址: kotlinx-serialization官网 ,它由两个主要部分组成:

  • Gradle插件org.jetbrains.kotlin.plugin.serialization

  • 运行时库

Kotlinx-Serialization由Kotlin语言编写,并且利用了Kotlin编译器能够生成代码的特性从而为可序列化的类生成序列化反序列化相关的代码,从而不需要向其它框架一样使用反射,另外一个特点就是它的体积很小,并且对数据类进行了很好的支持,并且API也和其它JSON解析框架一样丰富,能够满足我们所需。

1、kotlin-serialization的环境配置


导入依赖:

implementation "org.jetbrains.kotlinx:kotlinx-serialization-json:1.5.1"

开启编译器插件:我的测试项目Gradle版本是7.1.2,不同的版本配置略有差异,但目的都是配置插件。

模块的build.gradle:

plugins {id 'com.android.application'id 'org.jetbrains.kotlin.android'id 'org.jetbrains.kotlin.plugin.serialization' version '1.8.21'
}

工程的settings.gradle:

pluginManagement {repositories {gradlePluginPortal()google()mavenCentral()maven ()}plugins {id 'com.android.application' version '7.1.2'id 'com.android.library' version '7.1.2'id 'org.jetbrains.kotlin.android' version '1.8.21'id 'org.jetbrains.kotlin.plugin.serialization' version '1.8.21'}
}

这样插件就配置好了,接下来sync以下就可以使用kotlin官方的序列化了。

2、kotlin-serialization的基础介绍


在使用kotlin-serialization时,需要给数据类加上注解@Serializable,以此来标识这是一个可以被序列化的Kotlin类,从而编译器会生成对应的代码。

@kotlinx.serialization.Serializable
data class ComplexEntity(val id: Int,val name: String,val score: Float,val userInfo: UserInfo,val default: Int = 1223
)@kotlinx.serialization.Serializable
data class UserInfo(val userName: String,val userAge: Int,val userFriends: List<FriendsInfo>
)@kotlinx.serialization.Serializable
data class FriendsInfo(val phoneNumber: String,val userTag: String,val groupId: Int
)

通过字节码反编译后,我们看到编译器生成了一个内部类,它实现了GeneratedSerializer,这个接口是给编译器插件使用的,该内部类主要就是为Kotlin序列化提供相关的方法,将对象序列化为Json或者将Json字符串反序列化为对象都要通过该类的方法去实现。

序列化与反序列化示例:

private fun testFastJsonKtBean() {val complexEntity = ComplexEntity(100, "null", 99.90f, UserInfo("Tom", 24, mutableListOf<FriendsInfo>().apply {add(FriendsInfo("12311111111", "lover", 1))add(FriendsInfo("12322222222", "normal", 2))add(FriendsInfo("12333333333", "normal", 2))}))val jsonString = Json.encodeToString(complexEntity)Log.d(TAG, jsonString)val json = "{\"id\":100,\"name\":\"Android Developer\",\"score\":99.9,\"userInfo\":{\"userName\":\"Tom\",\"userAge\":24,\"userFriends\":[{\"phoneNumber\":\"12311111111\",\"userTag\":\"lover\",\"groupId\":1},{\"phoneNumber\":\"12322222222\",\"userTag\":\"normal\",\"groupId\":2},{\"phoneNumber\":\"12333333333\",\"userTag\":\"normal\",\"groupId\":2}]}}"val parseObject = Json.decodeFromString<ComplexEntity>(json)Log.d(TAG, parseObject.toString())
}

kotlin-serialization完全符合Kotlin属性的空安全性,解析的过程中对默认值也没有任何的影响,但是JSON字符串中常常会含有null或者空对象{}来表示某个属性为空,因此我们需要将可能出现为空的属性设置上默认值或者声明为可空类型来保证正常解析。

3、kotlin-serialization常用API


  • 序列化与反序列化:

//序列化JSON
Json.encodeToString(value:T):String
//JSON反序列化
Json.decodeFromString(string: String):T
  • 将JSON字符串转化为JsonElement,对应其它框架的JsonObject:

//该方法返回一个JsonElemnt
parseToJsonElement(json:String)
//判断jsonElement中是否含有key
jsonElement?.jsonObject?.containsKey(key:String)
//获取jsonElement
jsonElement?.jsonObject?.get(key)//获取key对应的各种类型的值
jsonElement?.jsonObject?.get(key)?.jsonPrimitive?.booleanjsonElement?.jsonObject?.get(key)?.jsonPrimitive?.intjsonElement?.jsonObject?.get(key)?.jsonPrimitive?.longjsonElement?.jsonObject?.get(key)?.jsonPrimitive?.floatjsonElement?.jsonObject?.get(key)?.jsonPrimitive?.doublejsonElement?.jsonObject?.get(key)?.jsonArray
  • 构造JsonObject:

val element = buildJsonObject {put("name", "kotlinx.serialization")putJsonObject("owner") {put("name", "kotlin")}putJsonArray("forks") {addJsonObject {put("votes", 42)}addJsonObject {put("votes", 9000)}}
}
  • 指定序列化和反序列化相关的Json配置:

//指定配置
val json = Json {//忽略Json字符串中冗余的键值对,否则会发生异常ignoreUnknownKeys = true//Json串中为null但是属性为no-null类型,或者枚举值不存在coerceInputValues = true
}

4、如何封装以适应现有业务


首先,为什么要对第三方框架进行封装呢?做过第三方框架升级的朋友应该很清楚,当我们在项目当中直接使用大量的第三方提供的库时,特别是某些框架对外提供的接口比较多的情况下,每个人在使用的时候写法不同,所使用的方法也不同,因此升级后,需要对每一处调用的地方进行Review,防止兼容性问题导致功能异常。因此最好的解决方式就是根据自己的业务对三方框架进行二次封装,这样在升级框架的时候,如果某些Api变化较大,我们只需要对自己封装的框架进行简单的改动就行。所以我们要针对于数据类解析对kotlin-serialization进行封装。

我们需要明白在客户端场景下对Json的解析最常用的Api是哪些,主要分为两大类,序列化和反序列化,前者用的相对没有反序列化频繁,但也是必须的,我们先来看反序列化。反序列化需要将JSON解析为对象,当然也有解析字段在进行手动构造的对象的情况。因此我们定义了一些适合业务的接口,其中的第一个接口是通过类型将JSON字符串解析为对象,同时要保证每个方法的健壮性以及可维护性。

/*** 配置解析为对象的时候忽略Json字符串中冗余的键值对*/
val json = Json {ignoreUnknownKeys = truecoerceInputValues = true
}/*** 调用举例,EntityName为返回的类型* 当JSON字符串中的值为 null 时,可以使用两种方式选其一可解决异常:* ① 声明对应的属性为可控类型,赋值为null* ② 给对应字段赋值默认值* ```* val entity:EntityName = KJson.parseObject<EntityName>(jsonStr)* ```* @param jsonStr 需要解析的Json字符串* @return 返回解析后对应的对象,对象类需要使用注解 @Serializable*/
inline fun <reified T> parseObject(jsonStr: String?): T? {jsonStr?.let {return try {json.decodeFromString<T>(jsonStr)} catch (e: Exception) {null}}return null
}

在FastJson或者Gson中有JsonObject类,在kotlin-serialization中前面介绍到对应的是JsonElement类,因此我们也需要提供对应的方法,这两个方法会把对象或者JSON字符串解析为JsonElement对象。

/*** 将对象转换成JsonElement?,发生异常时返回null* @param obj 有 @Serializable 注解标注的Kotlin类* @return JsonElement?*/
inline fun <reified T> parseJsonElement(obj: T): JsonElement? {return try {Json.encodeToJsonElement(obj)} catch (e: Exception) {null}
}/*** 将Json字符串解析为JsonElement,类似与其它框架的JsonObject* @param jsonStr Json字符串* @return JsonElement?*/
fun parseJsonElement(jsonStr: String?): JsonElement? {jsonStr?.let {return try {Json.parseToJsonElement(jsonStr)} catch (e: Exception) {null}}return null
}

在现有业务中,我们使用FastJson在解析数据之前,会对基本的数据进行校验才会进行接下来的解析,类似于下面的代码,我们封装的API接口也需要提供这些基本的方法。

6b7a527868c499abdd99abd9f7a0136d.png
fun containsKey(jsonElement: JsonElement, key: String): Boolean {return jsonElement.jsonObject.containsKey(key)
}

当我们尝试封装一个判断JsonElement是否含有某个key时,需要调用JsonElement类的containsKey()方法,这样设计对业务调用者不利,需要传两个参数。仔细分析,内部用于判断的实际方法JsonElement中的jsonObject的判断方法,因此直接使用扩展函数对JsonElement进行扩展,对应的方法如下:

/*** 判断JsonElement中是否含有某一个Key-value* @param key 需要查找的关键词* @return Boolean*/
fun JsonElement.containsKey(key: String): Boolean {return jsonObject.containsKey(key)
}/***  从JsonElement中获取[key]对应String类型的值,未找到或异常返回默认值[defaultStr]* @param key 需要查找的关键词* @param defaultStr 默认值* @return String*/
fun JsonElement.getString(key: String): String? {return try {if (containsKey(key))jsonObject[key]?.jsonPrimitive?.contentOrNullelse null} catch (e: Exception) {null}
}/*** 从JsonElement中获取[key]对应Int类型的值,默认值为[defaultVal]* @param key 需要查找的关键词* @param defaultVal 不存在或异常情况返回默认值* @return Int*/
fun JsonElement.getInt(key: String, defaultVal: Int = 0): Int {return try {if (containsKey(key))jsonObject[key]?.jsonPrimitive?.int ?: defaultValelse defaultVal} catch (e: Exception) {defaultVal}
}/*** 从JsonElement中获取[key]对应的JsonArray,发生异常时返回null* @param key 需要查找的关键词* @return JsonArray?*/
fun JsonElement.getJsonArray(key: String): JsonArray? {return try {if (containsKey(key))jsonObject[key]?.jsonArrayelse null} catch (e: Exception) {null}
}

以上封装了获取String和Int类型以及JsonArray类型的API,其它基本数据类型的API类似,就不再赘述了。我们对反序列化的API进行了封装,接下来是对序列化的API进行一个封装:

/*** @param obj 需要转换的对象,对象类需要使用注解 @Serializable* @return 返回对象转换后的Json字符串*/
fun toJsonString(obj: Any): String? {return try {Json.encodeToString(obj)} catch (e: Exception) {printLog("toJsonString", e)null}
}

到此,业务常用的API及已经封装完成了,这样大家在使用的时候直接调用封装的接口,如果要升级我们只需要检验框架中的API是否会有兼容性的问题即可,以便于业务的快速迭代。


参考

  • Aoe, J. I. (1989). An efficient implementation of static string pattern matching machines. IEEE Transactions on SoftwareEngineering, 15(8), 1010-1016.

  • GitHub - Kotlin/kotlinx.serialization: Kotlin multiplatform / multi-format serialization

  • https://github.com/Kotlin/kotlinx.serialization/blob/master/docs/serialization-guide.md

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/650611.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CAD-autolisp(二)——选择集、命令行设置对话框、符号表

目录 一、选择集1.1 选择集的创建1.2 选择集的编辑1.3 操作选择集 二、命令行设置对话框2.1 设置图层2.2 加载线型2.3 设置字体样式2.4 设置标注样式&#xff08;了解即可&#xff09; 三、符号表3.1 简介3.2 符号表查找3.2 符号表删改增 一、选择集 定义&#xff1a;批量选择…

【广度优先搜索】【拓扑排序】【C++算法】913. 猫和老鼠

作者推荐 【动态规划】【map】【C算法】1289. 下降路径最小和 II 本文涉及知识点 广度优先搜索 拓扑排序 逆推 LeetCode913. 猫和老鼠 两位玩家分别扮演猫和老鼠&#xff0c;在一张 无向 图上进行游戏&#xff0c;两人轮流行动。 图的形式是&#xff1a;graph[a] 是一个列…

List使用addAll()方法报错

当使用Arrays.asList方式创建出来的list&#xff0c;在使用addAll方法的时候报错如下&#xff1a; Exception in thread "main" java.lang.UnsupportedOperationException 这个问题记录下&#xff0c;以防以后忘记。 下面是代码 List<String> objects new A…

风口抓猪-借助亚马逊云科技EC2服务器即刻构建PalWorld(幻兽帕鲁)私服~~~持续更新中

Pocketpair出品的生存类游戏《幻兽帕鲁》最近非常火&#xff0c;最高在线人数已逼近200万。官方服务器亚历山大&#xff0c;游戏开发商也提供了搭建私人专用服务器的方案&#xff0c;既可以保证稳定的游戏体验&#xff0c;也可以和朋友一起联机游戏&#xff0c;而且还能自定义经…

LeetCode:1706. 球会落何处(Java 模拟)

目录 1706. 球会落何处 题目描述&#xff1a; 实现代码与解析&#xff1a; 原理思路&#xff1a; 1706. 球会落何处 题目描述&#xff1a; 用一个大小为 m x n 的二维网格 grid 表示一个箱子。你有 n 颗球。箱子的顶部和底部都是开着的。 箱子中的每个单元格都有一个对角线…

如何实现无公网IP实现远程访问MongoDB文件数据库

&#x1f4d1;前言 本文主要是如何实现无公网IP实现远程访问MongoDB文件数据库的文章&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是青衿&#x1f947; ☁️博客首页&#xff1a;CSDN主页放风讲故事 &#x…

移动Web——平面转换-平移

1、平面转换-平移 取值 像素单位数值百分比&#xff08;参照盒子自身尺寸计算结果&#xff09;正负均可 技巧 translate()只写一个值&#xff0c;表示沿着X轴移动单独设置X或Y轴移动距离&#xff1a;translateX()或translateY() <!DOCTYPE html> <html lang"en&q…

53-JS之BOM,打开,关闭窗口,screen对象,history对象,location对象,工作区尺寸,滚动距离

1.简介 BOM(Browser Object Model)---浏览器对象模型,提供JS当中对浏览器的各种操作对象 1.1BOM与DOM 2.打开窗口window.open(URL,name,features) 2.1 URL字符串:地址网址文件源 2.2name:指target属性,规定在哪个窗口打开新的url链接 blank:打开一个新窗口 _parent…

MVC架构模式与三层架构

提示&#xff1a;博客中的图片来源于动力节点在B站的视频讲解。 MVC架构模式与三层架构 一、三层架构二、MVC&#xff08;model view controller&#xff09;1.MVC 架构的工作流程&#xff08;1&#xff09;JSP Servlet javabean实现MVC。&#xff08;2&#xff09;SSM&#…

Linux——文本编辑器Vim

Linux中的所有内容以文件形式管理&#xff0c;在命令行下更改文件内容&#xff0c;常常会用到文本编辑器。我们首选的文本编辑器是Vim&#xff0c;它是一个基于文本界面的编辑工具&#xff0c;使用简单且功能强大&#xff0c;更重要的是&#xff0c;Vim是所有Linux发行版本的默…

详解静态网页数据获取以及浏览器数据和网络数据交互流程

目录 前言 一、静态网页数据 二、网址通讯流程 1.DNS查询 2.建立连接 3.发送HTTP请求 4.服务器处理请求 5.服务器响应 6.渲染页面 7.页面交互 三、URL/POST/GET 1.URL 2.GET 形式 3.POST 形式 四.获取静态网页数据 前言 在网站设计领域&#xff0c;基于纯HTM…

机房及设备安全智慧监管AI+视频方案的设计和应用

一、背景分析 随着互联网的迅猛发展&#xff0c;机房及其配套设施的数量持续攀升&#xff0c;它们的运行状况对于企业运营效率和服务质量的影响日益显著。作为企业信息化的基石&#xff0c;机房的安全监测与管理的重要性不容忽视。它不仅关乎企业的稳定运营&#xff0c;同时也…

希尔排序-排序算法

前言 希尔排序固然很好&#xff0c;但是某些情况下&#xff0c;有很多缺点。例如下面这种情况&#xff1a; 9 之前的元素都已经有序&#xff0c;只有元素 1 和 2 的位置不对&#xff0c;使用插入排序几乎要移动整个数组的元素&#xff0c;效率很低。 这时候希尔排序横空出世&…

黑群晖显示真实的CPU型号和自定义CPU型号

黑群晖显示真实的CPU型号和自定义CPU型号 修改脚本执行脚本自定义显示的CPU型号 脚本的仓库地址:✈ 修改脚本 我的CPU型号是N100&#xff0c;这里因为架构识别有问题&#xff0c;所有CPU的型号后面会带一个UnKnown&#xff0c;感觉很别扭&#xff0c;所有修改了下脚本 if […

AcWing 2. 01背包问题(DP思想)

[题目概述] 有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。 第 i 件物品的体积是 v i v_i vi​&#xff0c;价值是 w i w_i wi​。 求解将哪些物品装入背包&#xff0c;可使这些物品的总体积不超过背包容量&#xff0c;且总价值最大。 输出最大价值。 输入格式 …

Vue组件之间的通信方式都有哪些

Vue组件之间的通信方式 组件间通信的概念组件间通信解决了什么组件间通信的分类 父子组件之间的通信兄弟组件之间的通信祖孙与后代组件之间的通信非关系组件间之间的通信 组件间通信的方案 props传递数据$emit 触发自定义事件refEventBusparent、rootattrs与listenersprovide …

Oracle分栏(非分页)查询

不知道Oracle怎么进行数据分栏(分栏: 因数据列过长, 部分数据作为新列显示). 在这里先记录一下粗浅的查询方法. 数据源例子: select 日用百货 as cat, 手电筒 as name, 20 as amount, 2024-01-27 as dt from dualunion allselect 餐饮美食 as cat, 鸡公煲 as name, 15.9 as amo…

SSEBop FEWS V6蒸散发ET年和月尺度数据分享

一、数据简介 SSEBOP FEWS V6是一个用于估算蒸散发&#xff08;evapotranspiration&#xff09;的模型或数据集。蒸散发是指地表和植物蒸发以及植物蒸腾的总和&#xff0c;是水循环中重要的组成部分。 SSEBOP FEWS V6是由美国地质调查局&#xff08;USGS&#xff09;开发的一…

Genome-wide association studies in R

全基因组关联&#xff08;GWA&#xff09;研究扫描整个物种基因组&#xff0c;寻找多达数百万个SNPs与特定感兴趣特征之间的关联。值得注意的是&#xff0c;感兴趣的性状实际上可以是归因于群体的任何类型的表型&#xff0c;无论是定性的&#xff08;例如疾病状态&#xff09;还…

支持IPv4与IPv6双协议栈的串口服务器,IPv6串口服务器

物联网是啥玩意儿&#xff1f;这是首先要搞明白的。按照百度百科的说法&#xff0c;是将各种信息传感设备&#xff0c;如射频识别&#xff08;RFID&#xff09;装置、红外感应器、全球定位系统、激光扫描器等种种装置与互联网结合起来而形成的一个巨大网络。这个说法有些复杂&a…