0

我在 Kotlin 中有一个简单的图书和借阅者图书馆的模型,如果图书有借阅者,则在其中签出图书。我使用箭头选项来编码借款人的缺席/存在:

data class Borrower(val name: Name, val maxBooks: MaxBooks)
data class Book(val title: String, val author: String, val borrower: Option<Borrower> = None)

我在将这些对象序列化/反序列化到 Gson 中的 JSON 时遇到问题 - 特别是在一本书Option<Borrower>中对 JSON的表示:null

[
  {
    "title": "Book100",
    "author": "Author100",
    "borrower": {
      "name": "Borrower100",
      "maxBooks": 100
    }
  },
  {
    "title": "Book200",
    "author": "Author200",
    "borrower": null
  }
]

我的反序列化代码:

fun jsonStringToBooks(jsonString: String): List<Book> {
    val gson = Gson()
    return try {
        gson.fromJson(jsonString, object : TypeToken<List<Book>>() {}.type)
    } catch (e: Exception) {
        emptyList()
    }
}

我得到一个空列表。几乎相同的jsonStringToBorrowers工作正常。

有人可以指出我正确的方向吗?

是否会使用不同的 JSON 库,kotlinx.serialization或者Klaxon是一个更好的主意,他们是如何做的null <-> None

谢谢!

4

1 回答 1

6

这个问题有点隐藏,因为您在返回空列表之前没有记录异常。如果您记录了该异常,您将得到以下信息:

java.lang.RuntimeException:无法调用没有参数的私有 arrow.core.Option()

这意味着 Gson 不知道如何创建一个Option类,因为它没有公共的空构造函数。事实上,Option是一个密封的类(因此是抽象的),有 2 个具体的子类:SomeNone。为了获得Option您的实例,应该使用其中一种工厂方法,例如Option.just(xxx)Option.empty()其他方法。

现在,为了修复您的代码,您需要告诉 Gson 如何反序列化一个Option类。为此,您需要为您的gson对象注册一个类型适配器。

一种可能的实现如下:

class OptionTypeAdapter<E>(private val adapter: TypeAdapter<E>) : TypeAdapter<Option<E>>() {

    @Throws(IOException::class)
    override fun write(out: JsonWriter, value: Option<E>) {
        when (value) {
            is Some -> adapter.write(out, value.t)
            is None -> out.nullValue()
        }
    }

    @Throws(IOException::class)
    override fun read(input: JsonReader): Option<E> {
        val peek = input.peek()
        return if (peek != JsonToken.NULL) {
            Option.just(adapter.read(input))
        } else {
            input.nextNull()
            Option.empty()
        }
    }

    companion object {

        fun getFactory() = object : TypeAdapterFactory {
            override fun <T> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
                val rawType = type.rawType as Class<*>
                if (rawType != Option::class.java) {
                    return null
                }
                val parameterizedType = type.type as ParameterizedType
                val actualType = parameterizedType.actualTypeArguments[0]
                val adapter = gson.getAdapter(TypeToken.get(actualType))
                return OptionTypeAdapter(adapter) as TypeAdapter<T>
            }
        }
    }

}

您可以通过以下方式使用它:

fun main(args: Array<String>) {
    val gson = GsonBuilder()
        .registerTypeAdapterFactory(OptionTypeAdapter.getFactory())
        .create()
    val result: List<Book> = try {
        gson.fromJson(json, TypeToken.getParameterized(List::class.java, Book::class.java).type)
    } catch (e: Exception) {
        e.printStackTrace()
        emptyList()
    }

    println(result)
}

该代码输出:

[Book(title=Book100, author=Author100, borrower=Some(Borrower(name=Borrower100, maxBooks=100))), Book(title=Book200, author=Author200, borrower=None)]
于 2018-11-03T13:50:48.650 回答