问题
业务上需求将一些数据缓存到本地,思路是界说个类,赋值后运用 Gson 转换为 Json 数据存到本地。但是由于需求 SpannableStringBuilder 来保存Text的富文本特点,测验序列化会 Json 后,再反序列化为 SpannableStringBuilder 赋值给 TextView 会有一些意外的错误。
Stack trace:
java.lang.IndexOutOfBoundsException: setSpan (0 ... -1) has end before start
at android.text.SpannableStringInternal.checkRange(SpannableStringInternal.java:485)
at android.text.SpannableStringInternal.setSpan(SpannableStringInternal.java:199)
at android.text.SpannableStringInternal.copySpansFromSpanned(SpannableStringInternal.java:87)
at android.text.SpannableStringInternal.<init>(SpannableStringInternal.java:48)
at android.text.SpannedString.<init>(SpannedString.java:35)
at android.text.SpannedString.<init>(SpannedString.java:44)
at android.text.TextUtils.stringOrSpannedString(TextUtils.java:532)
at android.widget.TextView.setText(TextView.java:6318)
at android.widget.TextView.setText(TextView.java:6227)
at android.widget.TextView.setText(TextView.java:6179)
探索
SpannableString
起先测验将 SpannableStringBuilder 转为 SpannableString:
val spannableStringBuilder = SpannableStringBuilder("测验文本")
val spannableString = SpannableString.valueOf(spannableStringBuilder)
尽管康复数据时不会报错,但 SpannableString 的特点悉数消失了。
Html
于是开始检索如何耐久化 SpannableStringBuilder, 在 Stackoverflow 上有这么一个计划
android: how to persistently store a Spanned?
其中提到需求能够运用 Android 的 Html 类的 Html.toHtml 办法将 SpannableStringBuilder 数据转换为 html 的标签语言,康复时再运用 Html.fromHtml
val spannableStringBuilder = SpannableStringBuilder("测验文本")
val htmlString = Html.toHtml(spannableStringBuilder)
val spannableStringBuilder = Html.fromHtml(htmlString)
测验了一个,以上方式确实是一个顺畅处理的崩溃问题。需求注意的是,Html 的两个办法都是耗时办法,最好异步调用。
自界说 Gson 序列化和反序列化适配器
项目的 Json 解析结构运用的是 Gson,支持自界说序列化和反序列化。于是,编写一个适配器完成 JsonSerializer 和 JsonDeserializer
class SpannableStringBuilderTypeAdapter : JsonSerializer<SpannableStringBuilder>,
JsonDeserializer<SpannableStringBuilder> {
override fun serialize(
src: SpannableStringBuilder?,
typeOfSrc: Type?,
context: JsonSerializationContext?
): JsonElement {
return src?.let {
JsonPrimitive(Html.toHtml(src))
} ?: JsonPrimitive("")
}
override fun deserialize(
json: JsonElement?,
typeOfT: Type?,
context: JsonDeserializationContext?
): SpannableStringBuilder {
return json?.let {
val fromHtml = Html.fromHtml(json.asString).trim()
SpannableStringBuilder(fromHtml)
} ?: SpannableStringBuilder("")
}
}
//运用
Gson gson = new GsonBuilder()
.setDateFormat("yyyy-MM-dd hh:mm:ss")
.registerTypeAdapter(SpannableStringBuilder.class,
new SpannableStringBuilderTypeAdapter())
.create();
以上代码能够很好的工作,如果仔细的话,能够注意到反序列化时用到 trim(),由于反序列化为 SpannableStringBuilder 后字符串末尾会多处两个换行符,这个 Stackoverflow 有提到HTML.fromHtml adds space at end of text?。
总结,这次探索让我对耐久化多了一些思路,对于一些无法修正源码的类能够自界说适配器来序列化。