Ответ 1
Вы можете использовать эту библиотеку https://github.com/cbeust/klaxon
Klaxon - это легкая библиотека для анализа JSON в Котлине.
Я получаю довольно глубокую объектную строку JSON из службы, которую я должен проанализировать для объекта JSON, а затем сопоставить его с классами.
Как преобразовать строку JSON в объект в Kotlin?
После этого сопоставление с соответствующими классами я использовал StdDeserializer из Jackson. Проблема возникает в тот момент, когда объект обладает свойствами, которые также должны быть десериализованы в классы. Я не смог получить объект mapper, по крайней мере, не знал, как внутри другого десериализатора.
Заранее благодарим за любую помощь. Предпочтительно, изначально я пытаюсь уменьшить количество зависимостей, которые мне нужны, поэтому, если ответ будет только для манипуляций JSON и разбора его, будет достаточно.
Вы можете использовать эту библиотеку https://github.com/cbeust/klaxon
Klaxon - это легкая библиотека для анализа JSON в Котлине.
Нет сомнений, что будущее синтаксического анализа в Kotlin будет за kotlinx.serialization. Это часть библиотек Котлина. Это все еще во время написания в стадии инкубатора.
https://github.com/Kotlin/kotlinx.serialization
import kotlinx.serialization.*
import kotlinx.serialization.json.JSON
@Serializable
data class MyModel(val a: Int, @Optional val b: String = "42")
fun main(args: Array<String>) {
// serializing objects
val jsonData = JSON.stringify(MyModel.serializer(), MyModel(42))
println(jsonData) // {"a": 42, "b": "42"}
// serializing lists
val jsonList = JSON.stringify(MyModel.serializer().list, listOf(MyModel(42)))
println(jsonList) // [{"a": 42, "b": "42"}]
// parsing data back
val obj = JSON.parse(MyModel.serializer(), """{"a":42}""")
println(obj) // MyModel(a=42, b="42")
}
Чтобы разобрать это:
val jsonString = """
{
"type":"Foo",
"data":[
{
"id":1,
"title":"Hello"
},
{
"id":2,
"title":"World"
}
]
}
"""
Используйте эти классы:
import org.json.JSONObject
class Response(json: String) : JSONObject(json) {
val type: String? = this.optString("type")
val data = this.optJSONArray("data")
?.let { 0.until(it.length()).map { i -> it.optJSONObject(i) } } // returns an array of JSONObject
?.map { Foo(it.toString()) } // transforms each JSONObject of the array into Foo
}
class Foo(json: String) : JSONObject(json) {
val id = this.optInt("id")
val title: String? = this.optString("title")
}
Использование:
val foos = Response(jsonString)
Не уверен, что это то, что вам нужно, но это то, как я это сделал.
Использование import org.json.JSONObject:
val jsonObj = JSONObject(json.substring(json.indexOf("{"), json.lastIndexOf("}") + 1))
val foodJson = jsonObj.getJSONArray("Foods")
for (i in 0..foodJson!!.length() - 1) {
val categories = FoodCategoryObject()
val name = foodJson.getJSONObject(i).getString("FoodName")
categories.name = name
}
Здесь пример json: { "Еда": { "FoodName": "Яблоки", "Вес": "110" } }
Вы можете использовать Gson
.
Шаг 1
Добавить компиляцию
compile 'com.google.code.gson:gson:2.8.2'
Шаг 2
Конвертировать JSON в Kotlin Bean
(используйте JsonToKotlinClass)
Как это
Данные Json
{
"timestamp": "2018-02-13 15:45:45",
"code": "OK",
"message": "user info",
"path": "/user/info",
"data": {
"userId": 8,
"avatar": "/uploads/image/20180115/1516009286213053126.jpeg",
"nickname": "",
"gender": 0,
"birthday": 1525968000000,
"age": 0,
"province": "",
"city": "",
"district": "",
"workStatus": "Student",
"userType": 0
},
"errorDetail": null
}
Kotlin Bean
class MineUserEntity {
data class MineUserInfo(
val timestamp: String,
val code: String,
val message: String,
val path: String,
val data: Data,
val errorDetail: Any
)
data class Data(
val userId: Int,
val avatar: String,
val nickname: String,
val gender: Int,
val birthday: Long,
val age: Int,
val province: String,
val city: String,
val district: String,
val workStatus: String,
val userType: Int
)
}
Шаг 3
Используйте Gson
var gson = Gson()
var mMineUserEntity = gson?.fromJson(response, MineUserEntity.MineUserInfo::class.java)
Я лично использую модуль Джексона для Kotlin, который вы можете найти здесь: jackson-module-kotlin.
implementation "com.fasterxml.jackson.module:jackson-module-kotlin:$version"
В качестве примера, вот код для анализа JSON дерева навыков Path of Exile, который является довольно тяжелым (84k строк при форматировании):
Код Котлина:
package util
import com.fasterxml.jackson.databind.DeserializationFeature
import com.fasterxml.jackson.module.kotlin.*
import java.io.File
data class SkillTreeData( val characterData: Map<String, CharacterData>, val groups: Map<String, Group>, val root: Root,
val nodes: List<Node>, val extraImages: Map<String, ExtraImage>, val min_x: Double,
val min_y: Double, val max_x: Double, val max_y: Double,
val assets: Map<String, Map<String, String>>, val constants: Constants, val imageRoot: String,
val skillSprites: SkillSprites, val imageZoomLevels: List<Int> )
data class CharacterData( val base_str: Int, val base_dex: Int, val base_int: Int )
data class Group( val x: Double, val y: Double, val oo: Map<String, Boolean>?, val n: List<Int> )
data class Root( val g: Int, val o: Int, val oidx: Int, val sa: Int, val da: Int, val ia: Int, val out: List<Int> )
data class Node( val id: Int, val icon: String, val ks: Boolean, val not: Boolean, val dn: String, val m: Boolean,
val isJewelSocket: Boolean, val isMultipleChoice: Boolean, val isMultipleChoiceOption: Boolean,
val passivePointsGranted: Int, val flavourText: List<String>?, val ascendancyName: String?,
val isAscendancyStart: Boolean?, val reminderText: List<String>?, val spc: List<Int>, val sd: List<String>,
val g: Int, val o: Int, val oidx: Int, val sa: Int, val da: Int, val ia: Int, val out: List<Int> )
data class ExtraImage( val x: Double, val y: Double, val image: String )
data class Constants( val classes: Map<String, Int>, val characterAttributes: Map<String, Int>,
val PSSCentreInnerRadius: Int )
data class SubSpriteCoords( val x: Int, val y: Int, val w: Int, val h: Int )
data class Sprite( val filename: String, val coords: Map<String, SubSpriteCoords> )
data class SkillSprites( val normalActive: List<Sprite>, val notableActive: List<Sprite>,
val keystoneActive: List<Sprite>, val normalInactive: List<Sprite>,
val notableInactive: List<Sprite>, val keystoneInactive: List<Sprite>,
val mastery: List<Sprite> )
private fun convert( jsonFile: File ) {
val mapper = jacksonObjectMapper()
mapper.configure( DeserializationFeature.ACCEPT_EMPTY_ARRAY_AS_NULL_OBJECT, true )
val skillTreeData = mapper.readValue<SkillTreeData>( jsonFile )
println("Conversion finished !")
}
fun main( args : Array<String> ) {
val jsonFile: File = File( """rawSkilltree.json""" )
convert( jsonFile )
JSON (без форматирования): http://filebin.ca/3B3reNQf3KXJ/rawSkilltree.json
Учитывая ваше описание, я считаю, что оно соответствует вашим потребностям.
Прежде всего.
Вы можете использовать плагин преобразования JSON в Kotlin Data class в Android Studio для отображения JSON в классы POJO (класс данных Kotlin). Этот плагин будет аннотировать ваш класс данных Kotlin в соответствии с JSON.
Затем вы можете использовать GSON конвертер для преобразования JSON в Kotlin.
Следуйте этому полному руководству: Kotlin Android Учебник по разбору JSON
Если вы хотите разобрать JSON вручную.
val **sampleJson** = """
[
{
"userId": 1,
"id": 1,
"title": "sunt aut facere repellat provident occaecati excepturi optio
reprehenderit",
"body": "quia et suscipit\nsuscipit recusandae consequuntur expedita"
}]
"""
Код для разбора выше JSON Array и его объекта с индексом 0.
var jsonArray = JSONArray(sampleJson)
for (jsonIndex in 0..(jsonArray.length() - 1)) {
Log.d("JSON", jsonArray.getJSONObject(jsonIndex).getString("title"))
}
Для конвертации JSON в Kotlin используйте http://www.json2kotlin.com/
Также вы можете использовать плагин Android Studio. Файл> Настройки, выберите Plugins
в левом дереве, нажмите "Обзор репозиториев...", найдите " JsonToKotlinClass ", выберите его и нажмите зеленую кнопку "Установить".
После перезагрузки AS вы можете использовать его. Вы можете создать класс с помощью File > New > JSON To Kotlin Class (JsonToKotlinClass)
. Другой способ - нажать Alt + K.
Затем вы увидите диалоговое окно для вставки JSON.
В 2018 году мне пришлось добавить package com.my.package_name
в начале класса.
http://www.jsonschema2pojo.org/ Привет, вы можете использовать этот сайт для конвертации json в pojo.
контроль + Alt + Shift + к
После этого вы можете вручную преобразовать этот класс модели в класс модели kotlin. с помощью вышеупомянутого ярлыка.
Немного поздно, но что угодно.
Если вы предпочитаете синтаксический анализ JSON JavaScript-подобным конструкциям, использующим семантику Kotlin, я рекомендую JSONKraken, автором которого я являюсь.
Предложения и мнения по этому вопросу высоко ценятся!
Загрузите источник дема отсюда (Разбор Json в android kotlin)
Добавьте эту зависимость:
compile 'com.squareup.okhttp3:okhttp:3.8.1'
Функция вызова api:
fun run(url: String) {
dialog.show()
val request = Request.Builder()
.url(url)
.build()
client.newCall(request).enqueue(object : Callback {
override fun onFailure(call: Call, e: IOException) {
dialog.dismiss()
}
override fun onResponse(call: Call, response: Response) {
var str_response = response.body()!!.string()
val json_contact:JSONObject = JSONObject(str_response)
var jsonarray_contacts:JSONArray= json_contact.getJSONArray("contacts")
var i:Int = 0
var size:Int = jsonarray_contacts.length()
al_details= ArrayList();
for (i in 0.. size-1) {
var json_objectdetail:JSONObject=jsonarray_contacts.getJSONObject(i)
var model:Model= Model();
model.id=json_objectdetail.getString("id")
model.name=json_objectdetail.getString("name")
model.email=json_objectdetail.getString("email")
model.address=json_objectdetail.getString("address")
model.gender=json_objectdetail.getString("gender")
al_details.add(model)
}
runOnUiThread {
//stuff that updates ui
val obj_adapter : CustomAdapter
obj_adapter = CustomAdapter(applicationContext,al_details)
lv_details.adapter=obj_adapter
}
dialog.dismiss()
}
})