问题

业务上需求将一些数据缓存到本地,思路是界说个类,赋值后运用 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?。

总结,这次探索让我对耐久化多了一些思路,对于一些无法修正源码的类能够自界说适配器来序列化。