当前位置:首页 > 后端开发 > 正文内容

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

邻居的猫1个月前 (12-09)后端开发395

前语

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
大众号:俗人程序猿
本文版权归作者一切,欢迎转载,但有必要保存此段声明,并给出原文链接,谢谢合作!

扫描二维码推送至手机访问。

版权声明:本文由51Blog发布,如需转载请注明出处。

本文链接:https://www.51blog.vip/?id=216

分享给朋友:

“iOS开发 - Swift Codable协议实战:快速、简略、高效地完结JSON和Model转化!” 的相关文章

java面试宝典,java官网

java面试宝典,java官网

1. JavaGuide 这是一个全面的Java学习与面试指南,涵盖了Java基础、集合、IO、并发、JVM、新特性等多方面的知识。非常适合准备Java面试的朋友使用。 2. 2024最全Java面试八股文 这篇文章分享了一套详细的Java面试手册,涵盖了MyBatis、Zooke...

Python网址,python官网免费下载

Python网址,python官网免费下载

以下是关于Python的一些重要网址信息,包括官方网站、教程网站和社区网站: Python 官方网站 Python 官方网站: 提供Python源代码和安装程序下载,最新版本为Python 3.13.1。 包含Python标准库的文档、教程和指南,可以在线获取。 Python 教程网站 Py...

php一句话,php官网

请提供具体的上下文或问题,以便我能提供相关的PHP代码示例。深入解析PHP一句话木马:原理、构造与免杀技巧一、PHP一句话木马原理PHP一句话木马,顾名思义,就是只需要一行代码就能实现攻击目的的木马。其核心原理是利用PHP中的eval()函数。eval()函数可以将字符串当作PHP代码执行,从而实现...

go英语怎么读,Go英语单词的正确发音与用法解析

1. 动词“去”(to go): 作为一般现在时,主语是第三人称单数时(如 he she it),读音为 /g?/。 其他情况下,读音为 /go?/。2. 名词“围棋”(a board game): 在这个词组中,go 读音为 /ɡo?/。3. 名词“能,行”(permission...

python定义一个变量,Python变量定义详解

python定义一个变量,Python变量定义详解

我已经定义了一个名为 `my_variable` 的变量,其值为 42。现在这个变量的值是 42。Python变量定义详解在Python编程语言中,变量是存储数据的基本单元。理解如何定义和使用变量对于编写有效的Python代码至关重要。本文将详细介绍Python中变量的定义方法、规则以及一些实用的技...

java连接数据库

java连接数据库

Java连接数据库通常涉及以下几个步骤:1. 加载数据库驱动:首先需要加载数据库的驱动程序。这通常通过调用 `Class.forName` 方法完成。2. 建立数据库连接:使用 `DriverManager.getConnection` 方法来建立与数据库的连接。你需要提供数据库的URL、用户名和密...