千呼万唤始出来,Kotlin官方序列化库终相见(一)

*/

val lastName by lazy {

name.split(" ")[1]

}

}

正是因为by lazy跟随的属性可以在运行时算出来,所以序列化的时候他们会被忽略从而减小Json长度。因为延迟初始化属性在对象生成的时候只是一个空引用,Gson从json字符串取回的对象相应属性也是null,Gson把KClass当作JavaClass对待,延迟执行的代码信息也丢了。 如果你一定要既用Gson又要延迟初始化, 百度搜索“@Poko“了解详情。

主角是kotlinx.serialization 首先配置Gradle:

//build.gradle

plugins {

id ‘org.jetbrains.kotlin.jvm’ version ‘1.4.20’

id ‘org.jetbrains.kotlin.plugin.serialization’ version ‘1.4.20’

}

repositories {

// Artifacts are also available on Maven Central

jcenter()

}

dependencies {

implementation “org.jetbrains.kotlinx:kotlinx-serialization-json:1.0.1”

}

对于JSON,我们使用Json.encodeToString扩展功能对数据进行编码。它将可序列化的对象作为其参数在后台进行序列化,并将其编码为JSON字符串。 让我们从描述项目的类开始,并尝试获取其JSON表示形式。

@Serializable

data class Project(val name: String, val language: String)

fun main() {

val data = Project(“kotlinx.serialization”, “Kotlin”)

println(Json.encodeToString(data))

//打印{“name”:“kotlinx.serialization”,“language”:“Kotlin”}

val data = Json.decodeFromString(“”"

{“name”:“kotlinx.serialization”,“language”:“Kotlin”}

“”")

println(data)//Project(name=kotlinx.serialization, language=Kotlin)

}

再次提醒,并不是只有数据类才能序列化,只是为了反序列化时能把类直接打印出来。 敲黑板,@Serializable,可以加参数指定我们自定义的序列化器,无参数时使用系统给的Serializer。 幕后字段序列化 仅对有后备字段的类的属性进行序列化,因此具有getter / setter但却没有幕后字段的属性不会被序列化,被委托的属性也不会被序列化。

@Serializable

class Project(

// name is a property with backing field – serialized

var name: String

) {

var stars: Int = 0 // property with a backing field – serializedval

path: String // getter only, no backing field – not serialized

get() = “kotlin/$name”

var id by ::name // delegated property – not serialized

}

fun main() {

val data = Project(“kotlinx.serialization”).apply { stars = 9000 }

println(Json.encodeToString(data))

//{“name”:“kotlinx.serialization”,“stars”:9000}

}

如果我们想定义Project类,使其采用路径字符串,然后将其解构为相应的属性,则我们可能很想编写类似以下代码的内容:

@Serializable

class Project(path: String) {

val owner: String = path.substringBefore(‘/’)

val name: String = path.substringAfter(‘/’)

}

此类无法编译,因为@Serializable注解要求该类的主构造函数的所有参数均为属性。一个简单的解决方法是使用类的属性定义一个私有的主构造函数,然后将所需的构造函数转换为辅助构造函数。

@Serializable

class Project private constructor(val owner: String, val name: String) {

constructor(path: String) : this(

owner = path.substringBefore(‘/’),

name = path.substringAfter(‘/’)

)

val path: String

get() = “ o w n e r / owner/ owner/name”

}

fun main() {

println(Json.encodeToString(Project(“kotlin/kotlinx.serialization”)))

//{“owner”:“kotlin”,“name”:“kotlinx.serialization”}

}

path不具有幕后字段,不会被序列化。 数据验证 另一种情况是你可能想引入不带属性的主构造函数参数,在将其值存储到属性之前对其进行验证。为了使其可序列化,应该在主构造函数中将其替换为属性,然后将验证移至init {…}块中:

@Serializable

class Project(val name: String) {

init {

require(name.isNotEmpty()) { “name cannot be empty” }

}

}

fun main() {

val data = Json.decodeFromString(“”"

{“name”:“”}

“”")//Exception in thread “main” java.lang.IllegalArgumentException: name cannot be empty

println(data)

}

默认值

默认值属性反序列化时会被自动填充,序列化时不会被写入json,目的还是节省空间和带宽,在大多数实际场景中,此类配置可以减少视觉混乱,并节省要序列化的数据量。

0@Serializable

data class Project(val name: String, val language: String = “Kotlin”)

fun main() {

val data = Json.decodeFromString(“”"

{“name”:“kotlinx.serialization”}

“”")

println(data)//Project(name=kotlinx.serialization, language=Kotlin)

val data1 = Project(“kotlinx.serialization”)

println(Json.encodeToString(data1))//{“name”:“kotlinx.serialization”}

}

另一种类似情况是可空属性默认值为null

@Serializable

class Project(val name: String, val renamedTo: String? = null)

fun main() {

val data = Project(“kotlinx.serialization”)

println(Json.encodeToString(data))//{“name”:“kotlinx.serialization”}

}

当输入中存在可选属性时,该属性的相应初始化器不会调用。此功能是为提高性能而设计的,因此请注意不要依赖初始化程序中的副作用。

fun computeLanguage(): String {

println(“Computing”)

return “Kotlin”

}

@Serializable

data class Project(val name: String, val language: String = computeLanguage())

fun main() {

val data = Json.decodeFromString(“”"

{“name”:“kotlinx.serialization”,“language”:“Kotlin”}

“”")

println(data)//Project(name=kotlinx.serialization, language=Kotlin)

}

由于在输入中指定了language属性,因此在输出中看不到“计算”字符串。

@Required修饰的属性其值必须显式指明。

@Serializable

data class Project(val name: String, @Required val language: String = “Kotlin”)

fun main() {

val data = Json.decodeFromString(“”"

{“name”:“kotlinx.serialization”}

“”")

println(data)//Exception in thread “main” kotlinx.serialization.MissingFieldException: Field ‘language’ is required, but it was missing

}

@Transient修饰的属性不会被序列化,反序列化时也不能被指定。

@Serializable

data class Project(val name: String, @Transient val language: String = “Kotlin”)

fun main() {

val data = Json.decodeFromString(“”"

{“name”:“kotlinx.serialization”,“language”:“Kotlin”}

“”")/**

*Exception in thread “main” kotlinx.serialization.json.internal.JsonDecodingException:

*Unexpected JSON token at offset 60: Encountered an unknown key ‘language’.

*Use ‘ignoreUnknownKeys = true’ in ‘Json {}’ builder to ignore unknown keys.

*/

println(data)

}

Kt的序列化框架是严格支持Kt的类型系统的,所以下面的代码有异常:

@Serializable

data class Project(val name: String, val language: String = “Kotlin”)

fun main() {

val data = Json.decodeFromString(“”"

{“name”:“kotlinx.serialization”,“language”:null}

“”")//Exception in thread “main” kotlinx.serialization.json.internal.JsonDecodingException: Unexpected JSON token at offset 52: Expected string literal but ‘null’ literal was found.

//Use ‘coerceInputValues = true’ in 'Json {}` builder to coerce nulls to default values.

println(data)

}

套娃序列化

可序列化的类可以在其可序列化属性中引用其他类。被引用的类也必须标记为@Serializable

总结

最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的14套腾讯、字节跳动、阿里、百度等2021最新面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。

2020面试真题解析
腾讯面试真题解析

阿里巴巴面试真题解析

字节跳动面试真题解析
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
oid相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的14套腾讯、字节跳动、阿里、百度等2021最新面试真题解析,我把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。

[外链图片转存中…(img-7jXDUVE3-1715232237725)]
[外链图片转存中…(img-EH7Akacv-1715232237726)]

[外链图片转存中…(img-Sb4tUOG3-1715232237726)]

[外链图片转存中…(img-agu3thaL-1715232237727)]
网上学习 Android的资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值