From effce4621510149e66c990d5a75a20f951b6281a Mon Sep 17 00:00:00 2001 From: RicardoJiang <2868405029@qq.com> Date: Thu, 23 Nov 2023 10:46:09 +0800 Subject: [PATCH] Update Version and ReadeMe --- README.md | 48 ++++++++++++++--- README_zh.md | 51 ++++++++++++++++--- gradle.properties | 2 +- .../compiler/KudosFromJsonFunctionBuilder.kt | 3 +- 4 files changed, 86 insertions(+), 18 deletions(-) diff --git a/README.md b/README.md index 55ab2be..dfc3084 100644 --- a/README.md +++ b/README.md @@ -6,10 +6,10 @@ English | **[简体中文](README_zh.md)** # Kudos -**Kudos** is short for **K**otlin **u**tilities for **d**eserializing **o**bjects. It is designed to make it safer and easier to deserializing Kotlin classes with Gson and Jackson. +**Kudos** is short for **K**otlin **u**tilities for **d**eserializing **o**bjects. It is designed to make it safer and easier to deserializing Kotlin classes with Gson and Jackson. Moreover, it can also simplify the use of the high-performance deserialization framework, JsonReader. ## Background - +### Null Safety and Default Values When parsing JSON using common JSON serialization frameworks, Kotlin developers often face issues with no-arg constructors and property null safety. Let's illustrate these issues with some examples. ```kotlin @@ -99,6 +99,11 @@ Furthermore, Moshi is not always faster than Gson as benchmarks said. Most bench We kept thinking, is there a way to provide null safety and default values support of the primary constructor parameters for frameworks like Gson? The answer is **Kudos**. +### Performance +In actual use, it can be found that powerful frameworks like Gson, Moshi do not have great advantages in performance, while the system's native JsonReader has a clear edge in performance, especially during code cold start stages when it's not fully JIT optimized. However, JsonReader is costly to use, requiring developers to manually parse JSON, which is unfriendly for most developers. + +For this reason, Kudos provides a simple way for developers to use JsonReader through a customized compiler plugin, while ensuring the functionality of type null safety and primary constructor parameter default values. + ## Quick Start ### 1. Introduce Kudos into Gradle Projects @@ -163,6 +168,8 @@ kudos { gson = true // Enable Kudos.Jackson. Add kudos-jackson to dependencies. jackson = true + // Enable Kudos.AndroidJsonReader. Add kudos-android-json-reader to dependencies. + androidJsonReader = true } ```` @@ -323,6 +330,18 @@ println(user) // User(id=12, name=Benny Huo, age=-1, tel=) If the JSON lacks the `id` or `name` field, the parsing fails, ensuring that the properties of `User` is null safe. +The `Kudos` annotation also supports adding parameters like `KUDOS_GSON`, `KUDOS_JACKSON`, `KUDOS_ANDROID_JSON_READER`, to enable support for specific libraries, and also allows passing multiple parameters at the same time, for example: + +```kotlin +@Kudos(KUDOS_GSON, KUDOS_ANDROID_JSON_READER) +data class User( + val id: Long, + val name: String, + val age: Int = -1, + val tel: String = "" +) +``` + ### 4. Collections Properties of collection types declared in classes annotated with @Kudos, such as `List` or `Set`, will be handled carefully in the `validate` function to ensure the null safety for their elements. However, if the type to be parsed is like `List`, Kudos will not be able to provide null safety guarantees at runtime because it cannot obtain whether the element type is nullable. @@ -344,12 +363,25 @@ The code of nullability check has been optimized carefully which is generated by We also find out that Kudos.Gson performs better than Moshi on initializing. So it should be a better choice for low frequency deserialization scenarios with both performance(comparing to Moshi) and null safety(comparing to Gson). -| | small json | medium json | large json | -|---------------|----------------|----------------|----------------| -| Gson | 412,375 ns | 1,374,838 ns | 3,641,904 ns | -| Kudos-Gson | 517,123 ns | 1,686,568 ns | 4,311,910 ns | -| Jackson | 1,035,010 ns | 1,750,709 ns | 3,450,974 ns | -| Kudos-Jackson | 1,261,026 ns | 2,030,874 ns | 3,939,600 ns | +### Multi-run test results +| | small json | medium json | large json | +|------------------|----------------|----------------|----------------| +| Gson | 412,375 ns | 1,374,838 ns | 3,641,904 ns | +| Kudos-Gson | 517,123 ns | 1,686,568 ns | 4,311,910 ns | +| Jackson | 1,035,010 ns | 1,750,709 ns | 3,450,974 ns | +| Kudos-Jackson | 1,261,026 ns | 2,030,874 ns | 3,939,600 ns | +| JsonReader | 190,302 ns | 1,176,479 ns | 3,464,174 ns | +| Kudos-JsonReader | 215,974 ns | 1,359,587 ns | 4,019,024 ns | + +### One-run test results +| | small json | medium json | large json | +|------------------|-----------------|-----------------|-----------------| +| Gson | 3,974,219 ns | 4,666,927 ns | 8,271,355 ns | +| Kudos-Gson | 4,531,718 ns | 6,244,479 ns | 11,160,782 ns | +| Jackson | 12,821,094 ns | 13,930,625 ns | 15,989,791 ns | +| Kudos-Jackson | 13,233,750 ns | 15,674,010 ns | 18,641,302 ns | +| JsonReader | 662,032 ns | 2,056,666 ns | 4,624,687 ns | +| Kudos-JsonReader | 734,907 ns | 2,362,010 ns | 6,212,917 ns | For more detail, see [https://github.com/RicardoJiang/json-benchmark](https://github.com/RicardoJiang/json-benchmark) diff --git a/README_zh.md b/README_zh.md index 6888b59..a0d77b1 100644 --- a/README_zh.md +++ b/README_zh.md @@ -6,10 +6,10 @@ # Kudos -**Kudos** 是 **K**otlin **u**tilities for **d**eserializing **o**bjects 的缩写。它可以解决使用 Gson、Jackson 等框架反序列化 JSON 到 Kotlin 类时所存在的空安全问题和构造器默认值失效的问题。 +**Kudos** 是 **K**otlin **u**tilities for **d**eserializing **o**bjects 的缩写。它可以解决使用 Gson、Jackson 等框架反序列化 JSON 到 Kotlin 类时所存在的空安全问题和构造器默认值失效的问题,同时可以简化高性能的反序列化框架 JsonReader 的使用方式。 ## 问题背景 - +### 空安全与默认值问题 在使用常见的 JSON 序列化框架解析 JSON 时,Kotlin 开发者通常会面临无参构造器和属性空安全的问题。接下来我们通过举例来具体说明这几个问题。 ```kotlin @@ -99,6 +99,11 @@ User(id=12, name=Benny Huo, age=0, tel=null) 我们当时就一直在想,有没有什么办法为 Gson 这样的框架提供类型空安全和支持主构造器的参数默认值的能力呢?答案就是 **Kudos**。 +### 性能问题 +在实际使用中可以发现,Gson, Moshi 等功能强大的框架在性能上并没有很大优势,而系统原生的 JsonReader 在性能上却有着明显的优势,尤其是在冷启动阶段代码未经过充分 JIT 优化时。但是,JsonReader 的使用成本很高,需要开发者自己手动解析 JSON ,这对于大部分开发者来说是不友好的。 + +为此,Kudos 通过自定义编译器插件的方式,为开发者提供了一种简单的方式来使用 JsonReader,同时保证了类型空安全和主构造器参数默认值的功能。 + ## 快速上手 ### 1. 将 Kudos 集成到 Gradle 项目中 @@ -163,6 +168,8 @@ kudos { gson = true // 启用 Kudos.Jackson. 添加 kudos-jackson 依赖. jackson = true + // 启用 Kudos.AndroidJsonReader. 添加 kudos-android-json-reader 依赖. + androidJsonReader = true } ```` @@ -177,6 +184,9 @@ com.kanyun.kudos:kudos-gson // 仅当启用 Kudos.Jackson 时 com.kanyun.kudos:kudos-jackson + +// 仅当启用 Kudos.AndroidJsonReader 时 +com.kanyun.kudos:kudos-android-json-reader ``` 当然,开发者也可以在合适的场景下手动引入这些依赖。 @@ -322,6 +332,18 @@ println(user) // User(id=12, name=Benny Huo, age=-1, tel=) 如果 JSON 中缺少 id 或者 name 字段,则解析失败,确保 User 属性的类型空安全。 +`Kudos`注解也支持添加`KUDOS_GSON`, `KUDOS_JACKSON`, `KUDOS_ANDROID_JSON_READER`等参数, 开启对指定库的支持, 也支持同时传递多个参数,例如: + +```kotlin +@Kudos(KUDOS_GSON, KUDOS_ANDROID_JSON_READER) +data class User( + val id: Long, + val name: String, + val age: Int = -1, + val tel: String = "" +) +``` + ### 4. 集合类型的支持 被 `@Kudos` 标注的类的属性类型如果是集合类型,包括 `List`、`Set` 等,解析之后会在 `validate` 函数中校验元素是否为 `null` 来确保类型空安全。但如果要解析的类型是 `List`,Kudos 在运行时会因为无法获取到元素类型是否可空而无法提供类型空安全的保证。 @@ -343,12 +365,25 @@ val list = kudosGson().fromJson("""[null]""", typeOf>().javaType 在解析 JSON 时,考虑到冷启动的初始化耗时的情况,Kudos.Gson 比 Moshi 在大部分测试下性能更优(只有在多次解析同一数据类型时 Moshi 性能表现更好),因此 Kudos.Gson 在低频次的 JSON 解析场景下兼具了运行性能(优于 Moshi)和数据安全(优于 Gson)的优点。 -| | small json | medium json | large json | -|---------------|----------------|----------------|----------------| -| Gson | 412,375 ns | 1,374,838 ns | 3,641,904 ns | -| Kudos-Gson | 517,123 ns | 1,686,568 ns | 4,311,910 ns | -| Jackson | 1,035,010 ns | 1,750,709 ns | 3,450,974 ns | -| Kudos-Jackson | 1,261,026 ns | 2,030,874 ns | 3,939,600 ns | +### 多次运行测试结果 +| | small json | medium json | large json | +|------------------|----------------|----------------|----------------| +| Gson | 412,375 ns | 1,374,838 ns | 3,641,904 ns | +| Kudos-Gson | 517,123 ns | 1,686,568 ns | 4,311,910 ns | +| Jackson | 1,035,010 ns | 1,750,709 ns | 3,450,974 ns | +| Kudos-Jackson | 1,261,026 ns | 2,030,874 ns | 3,939,600 ns | +| JsonReader | 190,302 ns | 1,176,479 ns | 3,464,174 ns | +| Kudos-JsonReader | 215,974 ns | 1,359,587 ns | 4,019,024 ns | + +### 一次运行测试结果 +| | small json | medium json | large json | +|------------------|-----------------|-----------------|-----------------| +| Gson | 3,974,219 ns | 4,666,927 ns | 8,271,355 ns | +| Kudos-Gson | 4,531,718 ns | 6,244,479 ns | 11,160,782 ns | +| Jackson | 12,821,094 ns | 13,930,625 ns | 15,989,791 ns | +| Kudos-Jackson | 13,233,750 ns | 15,674,010 ns | 18,641,302 ns | +| JsonReader | 662,032 ns | 2,056,666 ns | 4,624,687 ns | +| Kudos-JsonReader | 734,907 ns | 2,362,010 ns | 6,212,917 ns | 更多细节可见:[https://github.com/RicardoJiang/json-benchmark](https://github.com/RicardoJiang/json-benchmark) diff --git a/gradle.properties b/gradle.properties index 9345f6b..d4ec88b 100644 --- a/gradle.properties +++ b/gradle.properties @@ -1,6 +1,6 @@ kotlin.code.style=official -VERSION_NAME=1.8.20-1.0.1 +VERSION_NAME=1.8.20-1.0.2 GROUP=com.kanyun.kudos diff --git a/kudos-compiler/src/main/java/com/kanyun/kudos/compiler/KudosFromJsonFunctionBuilder.kt b/kudos-compiler/src/main/java/com/kanyun/kudos/compiler/KudosFromJsonFunctionBuilder.kt index 840ae36..3779bab 100644 --- a/kudos-compiler/src/main/java/com/kanyun/kudos/compiler/KudosFromJsonFunctionBuilder.kt +++ b/kudos-compiler/src/main/java/com/kanyun/kudos/compiler/KudosFromJsonFunctionBuilder.kt @@ -56,6 +56,7 @@ import org.jetbrains.kotlin.ir.symbols.IrSimpleFunctionSymbol import org.jetbrains.kotlin.ir.types.IrSimpleType import org.jetbrains.kotlin.ir.types.IrType import org.jetbrains.kotlin.ir.types.IrTypeProjection +import org.jetbrains.kotlin.ir.types.classFqName import org.jetbrains.kotlin.ir.types.defaultType import org.jetbrains.kotlin.ir.types.isSubtypeOfClass import org.jetbrains.kotlin.ir.util.SYNTHETIC_OFFSET @@ -249,7 +250,7 @@ internal class KudosFromJsonFunctionBuilder( putValueArgument(1, getParameterizedType(field.type)) } } else { - throw Exception("Kudos UnSupported type") + throw Exception("Kudos UnSupported type ${field.type.classFqName}") } }