Kotlin Json 序列化1 - 基本功能實(shí)現(xiàn)

參考:

  1. https://github.com/yole/jkid

預(yù)備知識(shí)

  • Kotlin 反射;
  • Kotlin 注解;
  • Koltin 泛型;

說(shuō)明

Json 序列化是將一個(gè)對(duì)象,轉(zhuǎn)換成Json形式的字符串;反序列化是將json格式字符串轉(zhuǎn)成相應(yīng)的對(duì)象;

1. 分析

如下:代碼

// 數(shù)據(jù)類
data class Person(val name: String, val age: Int, val likes: List<String>)

// 測(cè)試代碼
val p = Person("better", 30, listOf("Java", "Kotlin", "Python"))
println(serialize(p)) 

輸出為:

{"age": 30, "likes": ["Java", "Kotlin", "Python"], "name": "better"}

將類的每個(gè)屬性作為 json 的 key,value 為 key 對(duì)象的值,如果value 為 List 形式,即對(duì)應(yīng)json的數(shù)組形式[];

我們可通過(guò)反射獲取類的所有屬性,然后遍歷,分別將每個(gè)屬性,映射成json對(duì)應(yīng)的 key 、value;

2. kotlin 代碼實(shí)現(xiàn)

會(huì)有大量字符串拼接,所以為StringBuilder添加擴(kuò)展函數(shù)來(lái)整體拼接

整體步驟:

  1. 獲取類的所有KProperty;
  2. 遍歷 KProperty 集合,獲取每個(gè) KProperty
  3. 每個(gè)KProperty都有他的名字與值,對(duì)應(yīng)json的 key,value,value 處理時(shí),需進(jìn)行類型判斷;
// 對(duì)外全局方法
fun serialize(obj: Any) = buildString { serializeObj(obj) }

private inline fun StringBuilder.serializeObj(o: Any) {
    // ==== 1. 獲取類的所有`KProperty`
    o.javaClass.kotlin.memberProperties.joinToStringBuilder(this, separator = ",", prefix = "{", postfix = "}") {
        // ==== 2.遍歷 `KProperty` 集合,獲取每個(gè) `KProperty`,進(jìn)行類型判斷,并處理;
        serializeProperty(it, o)
    }
}

private inline fun StringBuilder.serializeProperty(property: KProperty1<Any, *>, receiver: Any) {
    // ====3. 每個(gè)`KProperty`都有他的名字與值,對(duì)應(yīng)json的 key,value,value 處理時(shí),需進(jìn)行類型判斷;
    val key = property.name
    // 處理key
    serializeString(key)
    append(": ")

    // 處理value
    val value = property.get(receiver)
    serializePropertyValue(value)
}

// 處理屬性值
private fun StringBuilder.serializePropertyValue(value: Any?) {
    when (value) {
        null -> append("null")
        is String -> serializeString(value)
        is Boolean, is Number -> append(value.toString())
        is List<*> -> serializeList(value)
        else -> serializeObj(value)
    }
}

/**
 * 處理list
 */
private fun StringBuilder.serializeList(data: List<Any?>) {
    data.joinToStringBuilder(this,  separator = ", " ,prefix = "[", postfix = "]") {
        serializePropertyValue(it)
    }
}

// 屬性名稱
private inline fun StringBuilder.serializeString(name: String) {
    // like "better"
    append('\"').append("$name").append('\"')
}

// 這里對(duì)joinToString 進(jìn)行了改寫(xiě),使其支持 ((T) -> Unit)
fun <T> Iterable<T>.joinToStringBuilder(sb: StringBuilder, separator: CharSequence = ", ",
                                        prefix: CharSequence = "",
                                        postfix: CharSequence = "",
                                        limit: Int = -1,
                                        truncated: CharSequence = "...",
                                        transform: ((T) -> Unit)? = null): StringBuilder {
    return joinTo(sb, separator, prefix, postfix, limit, truncated) {
        if (transform == null) {    // 包裝器模式
            return@joinTo it.toString()
        }
        transform.invoke(it)
        ""
    }
}

3. 測(cè)試1 :

// 數(shù)據(jù)類
data class Person(val name: String, val age: Int, val likes: List<String>)

// 測(cè)試代碼
val p = Person("better", 30, listOf("Java", "Kotlin", "Python"))
println(serialize(p))  // 沒(méi)問(wèn)題

4. 測(cè)試2 ,新增類

// 縣
data class County(val name: String, val peopleCount: Int)
// 市
data class City(val name: String, val counties: List<County>)
// 省
data class Province(val name: String, val size: Int, val cities: List<City>?)

val c1 = County("湘潭縣", 20_000)
val c2 = County("株洲縣", 30_000)
val c4 = County("攸縣", 40_000)

val ci1 = City("湘潭市", listOf(c1))
val ci2 = City("株洲市", listOf(c2, c4))

val p1 = Province("湖南省", 780, listOf(ci1, ci2))
println(serialize(p1))

格式化為:

{
    "cities": [{
        "counties": [{
            "name": "湘潭縣",
            "peopleCount": 20000
        }],
        "name": "湘潭市"
    }, {
        "counties": [{
            "name": "株洲縣",
            "peopleCount": 30000
        }, {
            "name": "攸縣",
            "peopleCount": 40000
        }],
        "name": "株洲市"
    }],
    "name": "湖南省",
    "size": 780
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書(shū)系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Spring Cloud為開(kāi)發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,694評(píng)論 19 139
  • 前言 人生苦多,快來(lái) Kotlin ,快速學(xué)習(xí)Kotlin! 什么是Kotlin? Kotlin 是種靜態(tài)類型編程...
    任半生囂狂閱讀 26,733評(píng)論 9 118
  • 偶然得知就在昌平的深山溝里競(jìng)也藏著一處驚艷的‘’小貝加爾湖‘’!這就是長(zhǎng)峪城村北山上的小水庫(kù)。 長(zhǎng)峪城位于昌平西部...
    有時(shí)間_db72閱讀 510評(píng)論 0 0
  • SparkSQL結(jié)合SparkStreaming,使用SQL完成實(shí)時(shí)計(jì)算中的數(shù)據(jù)統(tǒng)計(jì) – lxw的大數(shù)據(jù)田地htt...
    葡萄喃喃囈語(yǔ)閱讀 7,190評(píng)論 0 2
  • 網(wǎng)絡(luò)上常常流傳漢語(yǔ)很【牛逼】的段子,大家看得不亦樂(lè)乎。但很多人其實(shí)是盲目跟風(fēng),瞎湊熱鬧。多數(shù)人由此產(chǎn)生了迷之優(yōu)越感...
    Fancyzzy閱讀 699評(píng)論 0 1

友情鏈接更多精彩內(nèi)容