iis服务器助手广告广告
返回顶部
首页 > 资讯 > 移动开发 >Android如何解析异构列表
  • 444
分享到

Android如何解析异构列表

2024-04-02 19:04:59 444人浏览 泡泡鱼
摘要

目录前言实现方案抽象封装总结前言 开发业务需求时,遇到了列表中包含完全不同类型的数据结构。这种列表我们称为异构列表。以聊天记录列表为例 [ { "msgType" : "

前言

开发业务需求时,遇到了列表中包含完全不同类型的数据结构。这种列表我们称为异构列表。以聊天记录列表为例


[
	{
		"msgType" : "text",
		"id" : "1",
		"content" : "Hello world"
	},
	{
		"msgType" : "record",
		"id" : "2",
		"url" : "https://xxxx.mp4",
		"length" : 123450
	},
	{
		"msgType" : "image",
		"id" : "3",
		"url" : "/file/imgs/upload/202211/11/kssyku0ayuv.jpg",
		"size" : "300x300"
	}
]

要想解析上面的JSON,手动解析不是不行,但肯定不推荐。如果直接使用解析工具,比如用Gson来解析的话,无论定义什么样的数据结构好像都不符合上面的列表元素。

那可不可以做到,我们告诉Gson列表中各个元素分别是什么样的数据类型,这样它不就知道该怎么解析了吗?接下来我们通过Gson的TypeAdapter来实现自定义解析。

实现方案

先定义好各种数据类型,与msgType字段一一对应


abstract class BaseMessage(val id: String?, val msgType: String?)

class TextMessage(id: String?, msgType: String?, val content: String?
) : BaseMessage(id, msgType)

class ImageMessage(id: String?, msgType: String?, val url: String?, val size: String?
) : BaseMessage(id, msgType)

class RecordMessage(id: String?, msgType: String?, val url: String?, val length: Long
) : BaseMessage(id, msgType)

接着自定义一个TypeAdapter。


class BaseMessageTypeAdapter : TypeAdapter<BaseMessage>() {
 override fun write(out: jsonWriter, value: BaseMessage?) {
 }

 override fun read(`in`: JsonReader): BaseMessage? {
 }
}

可以看到里面有两个方法:write()负责序列化,read()负责反序列化。我们先重点关注read()的实现

实现read()的基本思路如下

  1. 读取msgType字段
  2. 根据msgType判断对应的数据类型
  3. 根据该数据类型获取解析该类型的TypeAdapter
  4. 交给对应类型的TypeAdapter解析

依照上述思路,可以写出read()的基本实现代码。当然这是比较粗糙的实现,实际上还有其他情况要考虑


class BaseMessageTypeAdapter(private val gson: Gson, 
        private val skipPast: TypeAdapterFactory
) : TypeAdapter<BaseMessage>() {
 override fun read(`in`: JsonReader): BaseMessage? {
  // 1.读取msgType字段
  val jsonObject = Streams.parse(`in`).asJsonObject
  val msgType = jsonObject.get("msgType")?.asString
  // 2.根据msgType获取解析该类型的TypeAdapter
  val adapter = getTypeAdapterByType(msgType)
  // 3.交给对应类型的TypeAdapter解析
  return adapter?.fromJsonTree(jsonObject)
 }
}

write()方法没什么好说的,直接交给对应类型的TypeAdapter序列化


class BaseMessageTypeAdapter(private val gson: Gson, 
        private val skipPast: TypeAdapterFactory
) : TypeAdapter<BaseMessage>() {
 override fun write(out: JsonWriter, value: BaseMessage?) {
  if (value == null) {
   out.nullValue()
   return
  }
  getTypeAdapterByType(value.msgType)?.write(out, value)
 }
}

接着就是实现getTypeAdapterByType()方法。


 private fun getTypeAdapterByType(type: String?): TypeAdapter<BaseMessage>? {
  return when (type) {
   "text" -> getTypeAdapter(TextMessage::class.java)
   "image" -> getTypeAdapter(ImageMessage::class.java)
   "record" -> getTypeAdapter(RecordMessage::class.java)
   else -> null
  }
 }

 private fun <R : BaseMessage> getTypeAdapter(clazz: Class<R>): TypeAdapter<BaseMessage> {
  // 获取Gson中该类型对应的TypeAdapter
  return SubTypeAdapterWrapper(clazz, gson.getDelegateAdapter(skipPast, TypeToken.get(clazz)))
 }

逻辑也比较简单。需要注意的是,在getTypeAdapter()方法中,要将TypeAdapter<out BaseMessage>转换成TypeAdapter<BaseMessage>。接下来我们看看SubTypeAdapterWrapper是怎样实现的


class SubTypeAdapterWrapper<T, R : T>(private val clazz: Class<R>,
          private val adapter: TypeAdapter<R>
) : TypeAdapter<T>() {
 override fun write(out: JsonWriter, value: T) {
  if (!clazz.isInstance(value)) {
   throw JsonSyntaxException("Expected a " + clazz.name + " but was " + value)
  }
  adapter.write(out, value as R)
 }

 override fun read(`in`: JsonReader): T {
  return adapter.read(`in`)
 }
}

其实就是一个包装类。将解析R类型的TypeAdapter包装成解析T类型的TypeAdapter。

最后就是实现一个TypeAdapterFactory,并将其注册到Gson


class BaseMessageTypeAdapterFactory : TypeAdapterFactory {
 override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
  if (!BaseMessage::class.java.isAssignableFrom(type.rawType)) {
   return null
  }
  return BaseMessageTypeAdapter(gson, this) as (TypeAdapter<T>)
 }
}

写个测试用例测试一下

抽象封装

为了更好的复用,接下来我们要对BaseMessageTypeAdapter改造一下。

定义一个新的TypeAdapter子类。将类型和TypeAdapter的关系用Map来存储,并提供方法给外部调用。


public class HeterogeneousTypeAdapter<T> extends TypeAdapter<T> {
 private final Gson mGson;
 private final TypeAdapterFactory mSkipPast;
 private final String mFieldName;
 private final Map<String, TypeAdapter<T>> mClassToAdapterMap = new HashMap<>();
 private final Map<String, TypeAdapter<T>> mFieldToAdapterMap = new HashMap<>();

 public HeterogeneousTypeAdapter(Gson gson, TypeAdapterFactory skipPast, String fieldName) {
  mGson = gson;
  mSkipPast = skipPast;
  mFieldName = fieldName;
 }

 public <R extends T> void addSubTypeAdapter(final String fieldValue,
            final Class<R> cls) {
  final TypeAdapter<R> typeAdapter = mGson.getDelegateAdapter(mSkipPast, TypeToken.get(cls));
  addSubTypeAdapter(fieldValue, cls, typeAdapter);
 }

 public <R extends T> void addSubTypeAdapter(final String fieldValue,
            final Class<R> cls,
            final TypeAdapter<R> typeAdapter) {
  final TypeAdapter<T> adapter = new SubTypeAdapterWrapper<>(cls, typeAdapter);
  mClassToAdapterMap.put(cls.getName(), adapter);
  mFieldToAdapterMap.put(fieldValue, adapter);
 }

 @Override
 public void write(JsonWriter out, T value) throws IOException {
  if (value == null) {
   out.nullValue();
   return;
  }
  getTypeAdapterByClass(value.getClass()).write(out, value);
 }

 @Override
 public T read(JsonReader in) throws IOException {
  if (in.peek() == JsonToken.NULL) {
   in.nextNull();
   return null;
  }

  final JsonObject jsonObject = Streams.parse(in).getAsJsonObject();
  final JsonElement fieldElement = jsonObject.get(mFieldName);
  if (fieldElement == null || fieldElement.isJsonNull()) {
   throw new JsonSyntaxException("Field " + mFieldName + " is null or not found");
  }

  final String field = fieldElement.getAsJsonPrimitive().getAsString();
  final TypeAdapter<T> adapter = getTypeAdapterByField(field);
  if (adapter == null) {
   // Unknown field, just skip
   return null;
  }
  return adapter.fromJsonTree(jsonObject);
 }

 private TypeAdapter<T> getTypeAdapterByClass(Class<?> cls) {
  TypeAdapter<T> adapter = mClassToAdapterMap.get(cls.getName());
  if (adapter == null) {
   throw new JsonParseException("Unknown class : " + cls);
  }
  return adapter;
 }

 private TypeAdapter<T> getTypeAdapterByField(String field) {
  return mFieldToAdapterMap.get(field);
 }
}

使用方式


class BaseMessageTypeAdapterFactory : TypeAdapterFactory {
 override fun <T : Any?> create(gson: Gson, type: TypeToken<T>): TypeAdapter<T>? {
  if (!BaseMessage::class.java.isAssignableFrom(type.rawType)) {
   return null
  }
  val adapter = HeterogeneousTypeAdapter<BaseMessage>(gson, this, "msgType")
  // 注册各种类型
  adapter.addSubTypeAdapter("text", TextMessage::class.java)
  adapter.addSubTypeAdapter("image", ImageMessage::class.java)
  adapter.addSubTypeAdapter("record", RecordMessage::class.java)
  return adapter as (TypeAdapter<T>)
 }
}

总结

通过自定义TypeAdapter,我们实现了解析异构列表的功能。免除手动解析的繁琐工作,避免出现不必要的错误。

以上就是Android如何解析异构列表的详细内容,更多关于Android 解析异构列表的资料请关注编程网其它相关文章!

--结束END--

本文标题: Android如何解析异构列表

本文链接: https://www.lsjlt.com/news/121991.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Android如何解析异构列表
    目录前言实现方案抽象封装总结前言 开发业务需求时,遇到了列表中包含完全不同类型的数据结构。这种列表我们称为异构列表。以聊天记录列表为例 [ { "msgType" : "...
    99+
    2022-11-11
  • Python支持异步的列表解析式
    目录摘要基本原理和目标规范解析式中的 await语法的更新摘要 PEP-492 和 PEP-525 通过 async/await 语法,引入了对原生协程和异步生成器的支持。本 pep...
    99+
    2022-11-11
  • Android MVVM架构实现RecyclerView列表详解流程
    目录效果图导入引用导入Recyclerview依赖导入dataBinding引用代码解析建立实体类建立RecyclerView子项适配器建立适配器设置子项点击事件adapter全部代...
    99+
    2022-11-12
  • 如何解析python列表及数组
    本篇文章为大家展示了如何解析python列表及数组,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。列表List更像JAVA中的数组,但是其功能要比JAVA中的数组强大地多。有点类似于ArrayList...
    99+
    2023-06-17
  • python如何实现列表解析式
    这篇文章给大家分享的是有关python如何实现列表解析式的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。列表解析式Python中一个有用的特征是列表解析式。通过列表解析式,可以很方便地构造一个列表。列表解析式的一般...
    99+
    2023-06-27
  • Python支持异步的列表解析式是什么
    这篇文章主要讲解了“Python支持异步的列表解析式是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Python支持异步的列表解析式是什么”吧!摘要PEP-492 和 PEP-525 通...
    99+
    2023-07-02
  • PostgreSQL如何构建表达式解析
    这篇文章主要介绍PostgreSQL如何构建表达式解析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!一、数据结构EEO_XXX宏定义opcode分发器宏定义 #if def...
    99+
    2022-10-18
  • Python列表解析和生成器表达式的结构是什么
    今天小编给大家分享一下Python列表解析和生成器表达式的结构是什么的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下...
    99+
    2022-10-19
  • 如何分析python列表
    本篇文章为大家展示了如何分析python列表,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。1.列表python没有数组,而是引入了列表(list),列表可以存储任何类型的数据,而且同一个列表中的数据...
    99+
    2023-06-22
  • Android的RV列表刷新详解Payload与Diff方式异同
    目录前言一、Payload的刷新二、Diff的刷新与快速实现方法三、DiffUtil的封装小结前言 RecyclerView是我们常用的列表控件,一般来说当Item的数据改变的时候我...
    99+
    2022-11-13
    Android RV列表刷新Payload Diff Android RV 列表刷新
  • Android Material设计中列表和卡片的创建方法解析
    5.0提供了两个新的Widget,它们使用了Material Design 的style和animation: RecyclerView 一个更可插拔式的ListView,...
    99+
    2022-06-06
    列表 方法 material Android
  • Android Compose Column列表不自动刷新问题如何解决
    本篇内容介绍了“Android Compose Column列表不自动刷新问题如何解决”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能...
    99+
    2023-07-05
  • Android如何定义列表点击事件
    这篇文章将为大家详细讲解有关Android如何定义列表点击事件,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。列表点击事件的定义在Adapter内创建接口,并设置set方法// 声明接口inter...
    99+
    2023-06-22
  • Android Compose如何实现联系人列表
    这篇文章主要介绍“Android Compose如何实现联系人列表”,在日常操作中,相信很多人在Android Compose如何实现联系人列表问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答...
    99+
    2023-07-05
  • Android仿微信列表滑动删除 如何实现滑动列表SwipeListView
    接上一篇,本篇主要讲如何实现滑动列表SwipeListView。 上篇完成了滑动控件SwipeItemView,这个控件是一个自定义的ViewGroup,作为列表的一个ite...
    99+
    2022-06-06
    列表 Android
  • python列表构建器如何实现循环
    这篇文章主要介绍python列表构建器如何实现循环,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!python的数据类型有哪些python的数据类型:1. 数字类型,包括int(整型)、long(长整型)和float(...
    99+
    2023-06-14
  • Android Studio如何实现下拉列表效果
    这篇文章主要讲解了“Android Studio如何实现下拉列表效果”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Android Studio如何实现下拉列表效果”吧!题...
    99+
    2023-06-30
  • android RecycleView如何实现多级树形列表
    这篇文章给大家分享的是有关android RecycleView如何实现多级树形列表的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。本文实例为大家分享了android RecycleView实现多级树形列表的具体代...
    99+
    2023-06-15
  • knockoutjs模板如何实现树形结构列表
    小编给大家分享一下knockoutjs模板如何实现树形结构列表,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!数据结构  ...
    99+
    2022-10-19
  • 如何解析从PySpark中的字符串获取列表方法
    如何解析从PySpark中的字符串获取列表方法,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。在 PySpark 中是否有类似eval的功能。我正在尝试将 Python 代码...
    99+
    2023-06-22
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作