当前位置: 首页 > news >正文

HandyJSON原理

HandyJSON 的优势

JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式, 应用广泛. 在 App 的使用过程中, 服务端给移动端发送的大部分都是 JSON 数据, 移动端需要解析数据才能做进一步的处理. 在解析JSON数据这一块, 目前 Swift 中流行的框架基本上是 SwiftyJSON, ObjectMapper, JSONNeverDie, HandyJSON 这么几种.

我们应该如何选择呢?

首先我们应该先明白解析 JSON 的原理. 目前有两个方向.

保持 JSON 语义, 直接解析 JSON.
SwiftyJSON 就是这样的. 本质上仍然需要根据 JSON 结构去取值.

预定义 Model 类, 将 JSON 反序列化类的实例, 再使用这些实例.
这种方式和 OC 中的 MJExtension 的思路是一致的. 在 Swift 中, ObjectMapper, JSONNeverDie, 以及 HandyJSON 做的都是将 JSON 文本反序列化到 Model 类上.

第二种思路是我们熟悉和比较方便的. 和服务端定义好数据结构, 写好 Model 就可以直接解析.

第二种思路有三种实现方式:

完全沿用 OC 中的方式, 让 Model 类继承自 NSObject, 通过 class_copyPropertyList 方法拿到 Model 的各种属性, 从 JSON 中拿到对应的 value, 再通过 OC 中 利用runtime 机制 实现的 KVC 方法为属性赋值. 如 JSONNeverDie.
支持纯 Swift 类, 但要求开发者实现 mapping 函数, 使用重载的运算符进行赋值, 如 ObjectMapper.
获取到 JSON 数据后, 直接在内存中为实例的属性赋值. HandyJSON 就是这样实现的.
第一种方式的缺点在于需要强制继承 NSObject, 这不适用于 struct 定义的 Model. 因为 struct 创建的 Model 不能通过 KVC 为其赋值.
第二种方式的缺点在于自定义 mapping 函数, 维护比较困难.
第三种方式在使用上和 MJExtension 基本差不多, 比较方便. 是我们所推荐的.

HandyJSON 解析数据的原理.

如何在内存上为实例的属性赋值呢?
为属性赋值, 我们需要以下步骤:

获取到属性的名称和类型.
找到实例在内存中的 headPointer, 通过属性的类型计算内存中的偏移值, 确定属性在内存中的位置.
在内存中为属性赋值.
在 Swift 中实现反射机制的类是 Mirror, 通过 Mirror 类可以获取到类的属性, 但是不能为属性赋值, 它是可读的. 但是我们可以直接在内存中为实例赋值. 这是一种思路. 另外一种思路是不利用 Mirror, 直接在内存中获取到属性的名称和类型, 这也是可以的. 目前 HandyJSON 就是利用的第二种方式.

1. 核心原理:绕过反射,直接操作内存

传统的 JSON 库(如 ObjectMapper)依赖 Swift 的 Mirror 反射机制遍历模型属性,但反射存在性能瓶颈且无法直接修改属性值。HandyJSON 通过以下方式实现高效解析:

a. 利用类型元数据(Type Metadata)
  • Swift 编译器会为每个类型生成元数据,包含属性名称、类型、内存偏移量等信息。
  • HandyJSON 直接访问这些元数据,获取属性的名称内存位置,无需通过反射。
  • 例如,结构体的属性在内存中是连续排列的,通过元数据可以计算出每个属性的偏移量。
b. 内存拷贝与指针操作
  • 通过 UnsafeMutablePointer 直接操作模型实例的内存。

  • 将 JSON 值转换为目标类型后,直接写入对应的内存地址。

  • 示例代码逻辑:

    swift

    let offset = getPropertyOffset(from: metadata) // 获取属性偏移量
    let pointer = instancePointer + offset          // 计算属性内存地址
    let value = parseJSONValue(...)                // 解析 JSON 值
    pointer.storeBytes(of: value, as: type)         // 直接写入内存
    

2. 实现步骤详解

a. 类型元数据解析
  • 获取类型信息:通过 type(of:) 或泛型类型参数获取类型的元数据。
  • 解析属性列表:从元数据中提取属性名称、类型、是否为可选类型(Optional)等信息。
  • 处理继承和协议:遍历类型的继承链,确保父类属性也能被正确映射。
b. JSON 到内存的映射
  1. 解析 JSON:将 JSON 数据转换为字典([String: Any])。
  2. 匹配键与属性:将 JSON 的键与模型属性名匹配(支持自定义键名映射)。
  3. 类型转换:将 JSON 值转换为目标属性类型(如 String 转 Int、处理嵌套模型等)。
  4. 内存写入:通过指针将转换后的值写入模型实例的内存。
c. 特殊类型处理
  • 可选类型(Optional):根据 JSON 是否存在键值决定是否写入 nil
  • 枚举(Enum):将 JSON 值映射到枚举的 rawValue 或关联值。
  • 泛型类型:需要特殊处理元数据的动态解析。

相关文章:

  • softmax回归的实现
  • gma 2.1.3 (2025.03.19) 更新日志
  • 计算机考研复试机试-考前速记
  • flink广播算子Broadcast
  • Docker一键部署OpenObserve打造低成本的云原生观测平台操作详解
  • 操作系统知识点35
  • 基于MLA的人类语音情感分类
  • centos 7 部署FTP 服务用shell 脚本搭建
  • Git的安装
  • 代理IP与AI的碰撞:网络安全新防线解码
  • 【Java全栈进阶架构师实战:从设计模式到SpringCloudAlibaba,打造高可用系统】
  • 爬虫逆向解决debugger问题
  • 【QA】QT事件处理流程是怎么样的?
  • 如何理解前端工程化
  • 蓝桥杯备考:差分算法之 语文成绩
  • 信号的产生和保存
  • Netty源码—5.Pipeline和Handler一
  • Vue3+vite项目 使用require 解决 ReferenceError: require is not defined 报错问题
  • CodeBrick笔记
  • Jenkins集成Trivy安全漏洞检查指南
  • 讲座|为什么要不断地翻译叶芝的诗?它们为什么值得细读?
  • 医学统计专家童新元逝世,终年61岁
  • 总书记考察的上海“模速空间”,要打造什么样的“全球最大”?
  • 新造古镇丨上海古镇朱家角一年接待164万境外游客,凭啥?
  • 新经济与法|如何治理网购刷单与控评?数据合规管理是关键
  • 庆祝中华全国总工会成立100周年暨全国劳动模范和先进工作者表彰大会隆重举行,习近平发表重要讲话