我正在编写一个kotlin websocket,并想用gson将dtos转换为json。所以我写了一种包装器dto,其中包含真正的dto和一些关于dto的其他信息,比如它的类型。这是我的包装dto
class WrapperDto <T : AbstractDto> {
var type = ""
var action = ""
var dto : T = AbstractDto() as T
}
这是它可以包含的dtos之一:
class Person : AbstractDto() {
var firstName = ""
var familyName = ""
}
这里出于测试原因,我尝试将其转换为json并再次返回:
val wrapperDto2 = WrapperDto<Person>()
wrapperDto2.type = Person::class.simpleName!!;
wrapperDto2.action = "add"
val person = Person()
person.firstName = "Richard"
person.familyName = "Lederer"
wrapperDto2.dto = person;
val gson1 = Gson()
val toJson = gson1.toJson(wrapperDto2)
println("to json: " + toJson)
val fromJson = gson1.fromJson(toJson, wrapperDto2::class.java)
println("from json: " + fromJson)
永远不会调用最后一个println,我收到以下错误消息:
java.lang.IllegalArgumentException: Can not set at.richardlederer.contactmanager.dto.AbstractDto field at.richardlederer.contactmanager.dto.WrapperDto.dto to com.google.gson.internal.LinkedTreeMap
at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:167)
at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:171)
at sun.reflect.UnsafeObjectFieldAccessorImpl.set(UnsafeObjectFieldAccessorImpl.java:81)
我该如何解决这个问题?
答案 0 :(得分:3)
归因于type erasure的generics。您需要使用TokenType
在运行时检索类型信息。
val fromJson = gson1.fromJson<WrapperDto<Person>>(toJson, object: TypeToken<WrapperDto<Person>>() {}.type)
你也可以创建一个像这样的扩展函数:
inline fun <reified T> fromJson(json: String): T = Gson().fromJson<T>(json, object: TypeToken<T>() {}.type)
所以,你可以这样叫fromJson
:
val fromJson = fromJson<WrapperDto<Person>>(toJson)
//Or
val fromJson: WrapperDto<Person> = fromJson(toJson) //Type inferred