And*_*Dev 5 android json retrofit2 kotlinx.serialization
我将kotlinx.serialization与改造结合使用。我收到的 json 响应将根据它将包含的属性而有所不同。在大多数情况下,我的应用程序中的数据模型包含的字段比我在响应中收到的字段多。我无法控制它,所以我需要在代码中处理它。
Kotlinx.serializationMissingFieldException在这种情况下会抛出 a 。我知道在使用时Json.parse您可以将其包装在 try-catch 块中并忽略此类错误。但是由于我使用的是 Retrofit,所以我看不到使用这种方法的方法:
网络服务.kt
interface WebService {
@GET("person.json")
fun getPerson(): Call<MainActivity.Person>
}
Run Code Online (Sandbox Code Playgroud)
主活动.kt
class MainActivity : AppCompatActivity() {
@Serializable
data class Person(val name: String, val species: String, val missing: String)
@UnstableDefault
override fun onCreate(savedInstanceState: Bundle?) {
val mediaType = "application/json".toMediaTypeOrNull()
mediaType?.let {
retrofit = Retrofit.Builder()
.addConverterFactory(Json.nonstrict.asConverterFactory(it))
.baseUrl(baseUrl)
.build()
}
webService = retrofit.create(WebService::class.java)
GlobalScope.launch {
val person = fetchPerson(webService)
}
}
private suspend fun fetchPerson(webService: WebService): Person {
return suspendCancellableCoroutine { cont ->
webService.getPerson()
.enqueue(object : Callback<Person> {
override fun onFailure(call: Call<Person>, t: Throwable) {
Log.e(t.toString(), "Unable to get api response")
cont.cancel(t)
}
override fun onResponse(
call: Call<Person>,
response: Response<Person>
) {
if (response.isSuccessful) {
response.body()?.let { cont.resume(it) }
} else {
cont.cancel(IOException("${response.code()}: ${response.errorBody()}"))
}
}
})
}
}
}
Run Code Online (Sandbox Code Playgroud)
json 响应(在这个虚构的例子中)故意省略了 'missing' 字段:
{"name":"me", "species":"superhuman"}
Run Code Online (Sandbox Code Playgroud)
由于该 json 不包含missing来自数据类的字段,因此应用程序崩溃并抛出MissingFieldException. 我想知道如何在改造案例中避免这个问题。
谢谢你的帮助。
Md.*_*man 12
实际上它无法Person从 json创建对象,因为您的Person类构造函数需要 3 个值。您必须满足此要求才能创建对象。
解决此问题的一种可能解决方案是在Kotlin 中使用默认值,如下所示:
data class Person(
var name: String="",
var species: String="",
var missing: String="")
Run Code Online (Sandbox Code Playgroud)
另一种解决方案是使用多个具有不同参数的构造函数,但由于您提到它可能会随着时间的推移而有所不同,因此该解决方案可能不太方便。谢谢