前言

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

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

简单应用

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

struct GCPerson: Codable {
var name: String
var age: Int
var height: Float // cm
var 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 = jsonStr
print(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)

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

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

CodingKeys 字段映射

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

struct Address: Codable {
var zipCode: Int
var fullAddress: String enum 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 模型中的的 zipCodefullAddress 属性字段已被替换为 zip_codefull_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 型只支持 truefalse,其它一概不认。

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

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,因为这个会比上面的要常用。我们需将 JSONEncoderdateEncodingStrategy 设置为 secondsSince1970JSONDecoder 也是相同的设置。

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

struct Activity: Codable {
var time: Date
var 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\" / null
guard 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: String
var sex: Bool // 0/false女 1/true男 init(name: String, sex: Bool) {
self.name = name
self.sex = sex
} // 必须实现此枚举,在编码解码方法中需要用到
enum CodingKeys: CodingKey {
case name
case sex
} init(from decoder: Decoder) throws {
let container = try decoder.container(keyedBy: CodingKeys.self)
self.name = try container.decode(String.self, forKey: .name)
// 取出来int后再转换为Bool
let 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)

总结

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

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

Demo

我把代码放在了 github 上面,可以到这儿下载:GarveyCalvin/iOS-Travel

谢谢你这么好看还关注我,大家一起进步吧。

关于作者

博文作者:GarveyCalvin

公众号:凡人程序猿

本文版权归作者所有,欢迎转载,但必须保留此段声明,并给出原文链接,谢谢合作!

Swift Codable协议实战:快速、简单、高效地完成JSON和Model转换!的更多相关文章

  1. 《Android Studio实战 快速、高效地构建Android应用》--五、备忘录实验(1/2)

    通过开发App熟悉Android Studio的用法 开发一款用于管理备忘事项列表的App,核心功能: 创建.删除备忘 将某些备忘标记为重要(左侧带颜色标签突出显示) 涉及:操作栏菜单.上下文菜单.用 ...

  2. 《Android Studio实战 快速、高效地构建Android应用》--三、重构代码

    要成为高效的Android程序员,需要头脑灵活,能够在开发.调试和测试的过程中重构代码,重构代码最大的风险是可能会引入意外的错误,Android Studio通过分析某些具有危险性的重构操作来降低风险 ...

  3. 快速简单高效的搭建 SolrCloud 集群

    转https://segmentfault.com/a/1190000008634902 集群配置 集群中的每台机器都要按照以下说明进行配置启动 首先到 solr 安装目录的 bin 下,编辑 sol ...

  4. 《Android Studio实战 快速、高效地构建Android应用》--四、Git入门

    Git版本控制系统(VCS)是分布式的,仓库的每一个副本均包含项目的完整历史 安装Git 下载 下载地址:http://git-scm.com/downloads 选择适合自己操作系统的来下载 如果下 ...

  5. 《Android Studio实战 快速、高效地构建Android应用》--二、在Android Studio中编程

    代码折叠 Ctrl+数字加号展开光标处已折叠代码块 Ctrl+数字减号折叠光标处已展开代码块 Ctrl+Shift+数字加号展开窗口中全部代码 Ctrl+Shift+数字减号折叠窗口中全部代码 注释代 ...

  6. 《Android Studio实战 快速、高效地构建Android应用》--Android Studio操作

    前言 摩尔定律:CPU的处理能力大约18个月翻一倍 Android&Java:想要在Android Studio中开发Android App,必须以充分了解Java为前提(Java流行的原因: ...

  7. GitHub+jsDelivr+PicGo 打造稳定快速、高效免费图床

    标题: GitHub+jsDelivr+PicGo 打造稳定快速.高效免费图床 作者: 梦幻之心星 347369787@QQ.com 标签: [GitHub, 图床] 目录: 图床 日期: 2019- ...

  8. HiLink & LiteOS & IoT芯片 让IoT开发简单高效

    HiLink & LiteOS & IoT芯片让IoT开发简单高效 华为HiLink & LiteOS & IoT芯片使能三件套,让IoT开发更简单高效.下一代智能手机 ...

  9. 【中文分词】简单高效的MMSeg

    最近碰到一个分词匹配需求--给定一个关键词表,作为自定义分词词典,用户query文本分词后,是否有词落入这个自定义词典中?现有的大多数Java系的分词方案基本都支持添加自定义词典,但是却不支持HDFS ...

  10. 如何快速简单上传类库到CocoaPods - 图文攻略步骤

    当自己的库已经上传GitHub后,那么如何快速简单的开源自己的库呢? 这里就是介绍如何将自己的类库上传到pods管理库,以便开源所有人都能方便使用. 准备前提: - 项目已上传到GitHub (注意, ...

随机推荐

  1. Linux系列(8)-添加用户并设置密码

    #添加用户[root@iZm5ehnt0e8indgne1hibuZ ~]# useradd -m linsiyu #设置用户密码[root@iZm5ehnt0e8indgne1hibuZ ~]# p ...

  2. C# 自定义控件如何正确的继承父类

    C# 自定义控件可以分为三类: 复合控件:基本控件组合而成.应当继承自 UserControl 扩展控件:继承基本控件,扩展一些属性和事件.比如继承 Button 自定义控件:直接继承自 Contro ...

  3. 线上Java调优-Arthas入门

    1.SSH连接目标主机,找到对应容器ID docker ps | grep eam 2.进入容器,并启用bash docker exec -it 01c6ab243ff4 /bin/bash 3.按A ...

  4. dosbox debugger

    通过中断看程序运行过程 最终目的是要找到数据保存位置,如何保存到文件的.

  5. Promise静态方法实现(all race finally resolve reject)

    示例 // Promise.resolve() Promise.resolve(1).then((data) => { console.log(data) // 1 }) // Promise. ...

  6. 制作可以显示GIF动图的activeX 控件

    因为工作需要,我需要一个可以显示gif 动图的控件,用来在VBS中显示动图,结果找了半天发现根本没有这样的控件,所以只能搜集资料自己来制作一个. 下面记录一下步骤: 1. 下载 PictureEx.h ...

  7. ElasticSearch 实现分词全文检索 - term、terms查询

    数据准备 ElasticSearch 实现分词全文检索 - 测试数据准备 ElasticSearch的各种查询 不会对查询关键字进行分词 term 查询 term的查询是代表完全匹配,搜索之前不会对你 ...

  8. msfconsole的使用

    msfconsole是metasploit中的一个工具: msfconsole集成了很多漏洞的利用的脚本,并且使用起来很简单的网络安全工具 在终端输入msfconsole命令即可进入msf的控制台,m ...

  9. 痞子衡嵌入式:盘点国内RISC-V内核MCU厂商(2018年发布产品)

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是国内RISC-V内核MCU厂商(2018). 虽然RISC-V风潮已经吹了好几年,但2019年才是其真正进入主流市场的元年,最近国内大量 ...

  10. 能快速构建和定制网络拓扑图的WPF开源项目-NodeNetwork

    大家好,我是沙漠尽头的狼,今天介绍一个WPF开源项目-NodeNetwork,它可以帮助我们快速构建和定制网络拓扑图. 一.前言 在现代软件开发中,数据可视化和可交互性越来越受到关注.为了实现这一点, ...