iOS开发 - Swift Codable协议实战:快速、简单、高效地完成JSON和Model转换!

前言

Codable 是 Swift 4.0 引入的一种协议,它是一个组合协议,由 Decodable 和 Encodable 两个协议组成。它的作用是将模型对象转换为 JSON 或者是其它的数据格式,也可以反过来将 JSON 数据转换为模型对象。

Encodable 和 Decodable 分别定义了 encode(to:) 和 init(from:) 两个协议函数,分别用来实现数据模型的归档和外部数据的解析和实例化。最常用的场景就是刚提到的 JSON 数据与模型的相互转换,但是 Codable 的能力并不止于此。

简单应用

在实际开发中,Codable 的使用非常方便,只需要让模型遵循 Codable 协议即可:

struct GCPerson: Codable {var name: Stringvar age: Intvar height: Float // cmvar isGoodGrades: Bool
}

接下来编写数据编码和解码的方法:

func encodePerson() {let person = GCPerson(name: "XiaoMing", age: 16, height: 160.5, isGoodGrades: true)let encoder = JSONEncoder()encoder.outputFormatting = .prettyPrinted // 优雅永不过时,json会好看点哟do {let data = try encoder.encode(person)let jsonStr = String(data: data, encoding: .utf8)textView.text = jsonStrprint(jsonStr as Any)} catch let err {print("err", err)}
}func decodePerson() {let jsonStr = "{\"age\":16,\"isGoodGrades\":1,\"name\":\"XiaoMing\",\"height\":160.5}"guard let data = jsonStr.data(using: .utf8) else {print("get data fail")return}let decoder = JSONDecoder()do {let person = try decoder.decode(GCPerson.self, from: data)print(person)} catch let err {print("err", err)}
}

上面例子的输出:

Optional("{\n  \"age\" : 16,\n  \"isGoodGrades\" : true,\n  \"name\" : \"XiaoMing\",\n  \"height\" : 160.5\n}")
GCPerson(name: "XiaoMing", age: 16, height: 160.5, isGoodGrades: false)

应该有眼尖的童鞋是发现了,我将 JSONEncoder 的 outputFormatting 设置为了 prettyPrinted,这会让它输出的时候会美观一下,比如将它们放置在 UITextView 视图中作对比:

这里指的 default 是在没有设置 outputFormatting 的默认情况

CodingKeys 字段映射

如果属性名称与 JSON 数据中的键名不一致,需要使用 Swift 语言中的 CodingKeys 枚举来映射属性名称和键名。CodingKeys 是一个遵循了 CodingKey 协议的枚举,它可以用来描述 Swift 对象的属性与 JSON 数据中的键名之间的映射关系。

struct Address: Codable {var zipCode: Intvar fullAddress: Stringenum CodingKeys: String, CodingKey {case zipCode = "zip_code"case fullAddress = "full_address"}
}

数据编码和解码的方法与前面的大同小异:

func encodeAddress() {let address = Address(zipCode: 528000, fullAddress: "don't tell you")let encoder = JSONEncoder()encoder.outputFormatting = .prettyPrinted // 优雅永不过时,json会好看点哟do {let data = try encoder.encode(address)let jsonStr = String(data: data, encoding: .utf8)textView.text.append("\n\n")textView.text = textView.text.appending(jsonStr ?? "")print(jsonStr as Any)} catch let err {print("err", err)}
}func decodeAddress() {let jsonStr = "{\"zip_code\":528000,\"full_address\":\"don't tell you\"}"guard let data = jsonStr.data(using: .utf8) else {print("get data fail")return}let decoder = JSONDecoder()do {let address = try decoder.decode(Address.self, from: data)print(address)} catch let err {print("err", err)}
}

此时的输出为:

Optional("{\n  \"zip_code\" : 528000,\n  \"full_address\" : \"don\'t tell you\"\n}")
Address(zipCode: 528000, fullAddress: "don\'t tell you")

从控制台日志可以看出,Address 模型中的的 zipCode 和 fullAddress 属性字段已被替换为 zip_code 和 full_address,值得注意的是,使用 CodingKeys 映射后就只能使用映射后的字段名称。

数据类型匹配

Swift 中的数据类型需要与 JSON 数据中的数据类型匹配,否则将无法正确地进行解码。如果数据类型不匹配,则会进入到 catch 代码块,意味着解码失败。

let jsonStr = "{\"age\":16,\"isGoodGrades\":1,\"name\":\"XiaoMing\",\"height\":160.5}"

在上面的例子中,将 isGoodGrades 的值改为1,此时输出的错误内容为:

err typeMismatch(Swift.Bool, Swift.DecodingError.Context(codingPath: [CodingKeys(stringValue: "isGoodGrades", intValue: nil)], debugDescription: "Expected to decode Bool but found a number instead.", underlyingError: nil))

由此引出,Bool 型只支持 true 和 false,其它一概不认。

注意:只要是其中一个数据字段不能解析,则整条解析失败。

Date 和 Optional 可选类型

在使用 Codable 对 Date 和 Optional 属性进行编解码时,有些细节是需要了解的。

Codable 默认启用的时间策略是 deferredToDate,即从 UTC时间2001年1月1日0时0分0秒 开始的秒数,对应 Date 类型中 timeIntervalSinceReferenceDate 这个属性。比如 702804983.44863105 这个数字解析后的结果是 2023-04-10 07:34:17 +0000

在这儿把时间策略设置为 secondsSince1970,因为这个会比上面的要常用。我们需将 JSONEncoder 的 dateEncodingStrategy 设置为 secondsSince1970JSONDecoder 也是相同的设置。

在设置 Optional 可选类型时,在编码时,为空的属性不会包含在 JSON 数据中。在解码时,直接不传或将值设定为 \"null\" / \"nil\" / null 这三种值也能被解析为 nil

struct Activity: Codable {var time: Datevar url: URL?
}

编码解码的工作:

func encodeActivity() {let activity = Activity(time: Date(), url: URL(string: "https://www.baidu.com"))let encoder = JSONEncoder()encoder.outputFormatting = .prettyPrinted // 优雅永不过时,json会好看点哟encoder.dateEncodingStrategy = .secondsSince1970 // 秒do {let data = try encoder.encode(activity)let jsonStr = String(data: data, encoding: .utf8)textView.text.append("\n\n")textView.text = textView.text.appending(jsonStr ?? "")print(jsonStr as Any)} catch let err {print("err", err)}
}func decodeActivity() {
//        let jsonStr = "{\"time\":528000,\"url\":111}" // 即便是 Optional 的属性也要对应的数据类型,否则还是会解析失败let jsonStr = "{\"time\":1681055185}" // Optional类型的属性字段,直接不传也是nil//        let jsonStr = "{\"time\":528000,\"url\":null}" // 以下三种也能被解析为nil,\"null\" / \"nil\" / nullguard let data = jsonStr.data(using: .utf8) else {print("get data fail")return}let decoder = JSONDecoder()decoder.dateDecodingStrategy = .secondsSince1970 // 秒do {let activity = try decoder.decode(Activity.self, from: data)print(activity)} catch let err {print("err", err)}
}

此时的输出为:

Optional("{\n  \"url\" : \"https:\\/\\/www.baidu.com\",\n  \"time\" : 1681057020.835813\n}")
Activity(time: 2023-04-09 15:46:25 +0000, url: nil)

自定义编解码

有时候前后端定义的模型不同时,有可能会需要用到自定义编解码,以此来达成“统一”。

比如我们现在有一个 Dog 模型,sex 字段为 Bool 型,在后端的定义为 0 和 1,此时我们需要将它们给转换起来,可以是 false 为 0,true 为 1。

struct Dog: Codable {var name: Stringvar sex: Bool // 0/false女 1/true男init(name: String, sex: Bool) {self.name = nameself.sex = sex}// 必须实现此枚举,在编码解码方法中需要用到enum CodingKeys: CodingKey {case namecase sex}init(from decoder: Decoder) throws {let container = try decoder.container(keyedBy: CodingKeys.self)self.name = try container.decode(String.self, forKey: .name)// 取出来int后再转换为Boollet sexInt = try container.decode(Int.self, forKey: .sex)sex = sexInt == 1}func encode(to encoder: Encoder) throws {var container = encoder.container(keyedBy: CodingKeys.self)try container.encode(self.name, forKey: .name)// 将sex属性以int类型编码try container.encode(sex ? 1 : 0, forKey: .sex)}
}

在编码的时候将 sex 从 Bool 型转换为 Int 型,解码时则反过来。编解码的工作依旧与前面的大致一样:

func encodeDog() {let dog = Dog(name: "Max", sex: true)let encoder = JSONEncoder()encoder.outputFormatting = .prettyPrinted // 优雅永不过时,json会好看点哟do {let data = try encoder.encode(dog)let jsonStr = String(data: data, encoding: .utf8)textView.text.append("\n\n")textView.text = textView.text.appending(jsonStr ?? "")print(jsonStr as Any)} catch let err {print("err", err)}
}func decodeDog() {let jsonStr = "{\"name\":\"Max\",\"sex\":1}"guard let data = jsonStr.data(using: .utf8) else {print("get data fail")return}let decoder = JSONDecoder()do {let dog = try decoder.decode(Dog.self, from: data)print(dog)} catch let err {print("err", err)}
}

此时的日志输出为:

Optional("{\n  \"name\" : \"Max\",\n  \"sex\" : 1\n}")
Dog(name: "Max", sex: true)

总结

Codable 是 Swift 中非常方便的一个协议,可以帮助我们快速进行数据的编码和解码,提高了开发效率和代码可读性。当然使用不当也会造成严重的灾难,所以我为大家整理了以下几点使用时的注意事项,希望能对大家有所帮助:

  1. 嵌套的数据结构也需要遵循 Codable 协议。
  2. Bool 型只支持 true 或 false
  3. Optional 类型修饰的属性字段,直接不传是 nil,或将值设定为以下三种也能被解析为 nil\"null\" / \"nil\" / null
  4. 可以使用自定义的编码器和解码器来进行转换。 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/43886.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

web实现酷炫的canvas粒子动画背景

文章目录 前言一、particle-bg1. git地址:2. 安装3. 使用4. 完整demo 二、tsParticles1. 源码地址:2. 安装3. 引入4. 使用5. 几个例子5.1 ts粒子五彩纸屑烟花5.2 多粒子产卵器-用tsParticles制作5.3 ts粒子鼠标吸引力5.4 粒子烟花 源码地址完结 前言 粒…

【运筹优化】运输问题建模 + Java调用Cplex求解

文章目录 一、问题描述二、思路分析三、建模方案四、Java调用Cplex代码五、输出结果 一、问题描述 运输问题(transportation problem)一般是研究把某种商品从若干个产地运至若干个销地而使总运费最小的一类问题。 本博客将根据下面的例题,介绍运输问题…

STM32F407使用Helix库软解MP3并通过DAC输出,最精简的STM32+SD卡实现MP3播放器

只用STM32单片机SD卡耳机插座,实现播放MP3播放器! 看过很多STM32软解MP3的方案,即不通过类似VS1053之类的解码器芯片,直接用STM32和软件库解码MP3文件,通常使用了labmad或者Helix解码库实现,Helix相对labm…

WebRTC音视频通话-WebRTC视频自定义RTCVideoCapturer相机

WebRTC音视频通话-WebRTC视频自定义RTCVideoCapturer相机 在之前已经实现了WebRTC调用ossrs服务,实现直播视频通话功能。但是在使用过程中,RTCCameraVideoCapturer类提供的方法不能修改及调节相机的灯光等设置,那就需要自定义RTCVideoCaptur…

到江西赣州ibm维修服务器之旅-联想X3850 x6黄灯故障

2023年08月15日,一位江西赣州工厂客户通过朋友介绍与冠峰售前工程师取得联系,双方对产品故障前后原因沟通的大致情况如下: 服务器型号:Lenovo system x3850 x6 为用户公司erp仓库服务器 服务器故障:正常使用过程中业…

<数据结构与算法>二叉树堆的实现

目录 前言 一、树的概念及结构 1 树的概念 2 树的相关概念 二、二叉树的概念及结构 1.二叉树的概念 2. 特殊的二叉树 3. 二叉树的性质 4.二叉树的存储结构 三、二叉树的顺序结构及实现 1.堆的性质 2.堆的插入 3.堆的实现 堆的结构体 HeapInit 初始化 HeapPush 插入 HeapPop 删…

【C++进阶】继承、多态的详解(多态篇)

【C进阶】继承、多态的详解(多态篇) 目录 【C进阶】继承、多态的详解(多态篇)多态的概念多态的定义及实现多态的构成条件(重点)虚函数虚函数的重写(覆盖、一种接口继承)C11 override…

solr快速上手:聚合分组查询|嵌套分组指南(十二)

0. 引言 solr作为搜索引擎经常用于各类查询场景,我们之前讲解了solr的查询语法,而除了普通的查询语法,有时我们还需要实现聚合查询来统计一些指标,所以今天我们接着来查看solr的聚合查询语法 1. 常用聚合查询语法 以下演示我们…

面试题-React(一):React是什么?它的主要特点是什么?

探索React:前端开发中的重要角色与主要特点 引言: 在现代前端开发领域,React已经成为最受欢迎和广泛使用的JavaScript库之一。它由Facebook开发并于2013年首次发布。随着时间的推移,React在开发社区中获得了强大的支持和认可。本…

画质提升+带宽优化,小红书音视频团队端云结合超分落地实践

随着视频业务和短视频播放规模不断增长,小红书一直致力于研究:如何在保证提升用户体验质量的同时降低视频带宽成本? 在近日结束的音视频技术大会「LiveVideoStackCon 2023」上海站中,小红书音视频架构视频图像处理算法负责人剑寒向…

基于注意力神经网络的深度强化学习探索方法:ARiADNE

ARiADNE:A Reinforcement learning approach using Attention-based Deep Networks for Exploration 文章目录 ARiADNE:A Reinforcement learning approach using Attention-based Deep Networks for Exploration机器人自主探索(ARE)ARE的传统边界法非短视路径深度强化学习的方…

Python | Package | Python的三种包安装方式(pip/whl/tar.gz)

文章目录 PIP 安装与卸载Source 安装与卸载Whell 安装与卸载 PIP 安装与卸载 pip install xxx pip install xxxversion_numberpip install captcha pip install captcha0.4# XXX/anaconda3/envs/py373/lib/python3.7/site-packages pip uninstall captchaSource 安装与卸载 p…

C++音乐播放系统

C音乐播放系统 音乐的好处c发出声音乐谱与赫兹对照把歌打到c上 学习c的同学们都知道,c是一个一本正经的编程语言,因该没有人用它来做游戏、做病毒、做…做…做音乐播放系统吧!! 音乐的好处 提升情绪:音乐能够影响我们…

java语言B/S架构云HIS医院信息系统源码【springboot】

医院云HIS全称为基于云计算的医疗卫生信息系统( Cloud- Based Healthcare Information System),是运用云计算、大数据、物联网等新兴信息技术,按照现代医疗卫生管理要求,在一定区域范围内以数字化形式提供医疗卫生行业数据收集、存储、传递、…

动手学深度学习--基础知识上篇

🎈动手学deep learning ☁️本专栏会定期更新关于动手学深度学习的每章知识点的讲解,题目答案 👻如果喜欢,欢迎点赞,收藏 动手学深度学习-预备知识篇 线性代数篇 1-3题讲解 证明一个矩阵 A \mathbf{A} A的转置的转置…

安卓手机跑 vins slam (2)

既然选择把vins的代码移植到新工程,那么就需要先确定自己电脑的Android Studio的C开发环节是OK的,可以通过创建C的示例工程,能正常跑通做验证。 选择Native C 需要选择用C哪个版本, 这里通过百度搜索,slam 编译需要C 1…

电脑提示丢失(或找不到)msvcp120.dll解决办法

msvcp140.dll是Microsoft Visual C Redistributable的一部分,它是Windows操作系统中的一个动态链接库文件。这个文件包含了许多C标准库函数的实现,对于一些依赖C标准库的应用程序来说,msvcp140.dll是非常重要的。msvcp140.dll的主要用途是提供…

热电联产在综合能源系统中的选址定容研究(matlab代码)

目录 1 主要内容 目标函数 程序模型 2 部分代码 3 程序结果 1 主要内容 该程序参考《热电联产在区域综合能源系统中的定容选址研究》,主要针对电热综合能源系统进行优化,确定热电联产机组的位置和容量,程序以33节点电网和17节点热网为例…

CI/CD入门(二)

CI/CD入门(二) 目录 CI/CD入门(二) 1、代码上线方案 1.1 早期手动部署代码1.2 合理化上线方案1.3 大型企业上线制度和流程1.4 php程序代码上线的具体方案1.5 Java程序代码上线的具体方案1.6 代码上线解决方案注意事项2、理解持续集成、持续交付、持续部署 2.1 持续集成2.2 持续…

小白到运维工程师自学之路 第七十五集 (Kubernetes 企业级高可用部署)2

8、添加master节点 在k8s-master2和k8s-master3节点创建文件夹 mkdir -p /etc/kubernetes/pki/etcd在k8s-master1节点执行 从k8s-master1复制密钥和相关文件到k8s-master2和k8s-master3 scp /etc/kubernetes/admin.conf root192.168.77.15:/etc/kubernetes scp /etc/kubernet…