iOS開發 - Swift Codable協議實戰:快速、簡單、高效地完成JSON和Model轉換!

来源:https://www.cnblogs.com/GarveyCalvin/archive/2023/04/24/swift-codable-json-model.html
-Advertisement-
Play Games

Codable 是 Swift 4.0 引入的一種協議,它是一個組合協議,由 Decodable 和 Encodable 兩個協議組成。它的作用是將模型對象轉換為 JSON 或者是其它的數據格式,也可以反過來將 JSON 數據轉換為模型對象。 ...


前言

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
公眾號:凡人程式猿
本文版權歸作者所有,歡迎轉載,但必須保留此段聲明,並給出原文鏈接,謝謝合作!

活著,就是為了改變世界!
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 最近,在 CodePen 上,看到一個非常有意思的圖片動效,效果如下: 原效果鏈接:CodePen Demo - 1 div pure CSS blinds staggered animation in 13 declarations 本身這個動畫效果,並沒有多驚艷。驚艷的地方在於原作者的實現方式非 ...
  • HTML學習筆記詳解 01 初識HTML HTML HTML,英文全稱為 Hyper Text Markup Language,中文翻譯為超文本標記語言,其中超文本包括:文字,圖片,音頻,視頻,動畫等 目前 目前主流使用的是HTML5+CSS3 HTML的優勢 主流瀏覽器都支持 微軟 GOOGLE ...
  • 當用戶在網頁中進行操作時,如點擊、滾動、輸入等,往往會頻繁地觸發事件。如果每個事件都立即執行相應的函數,可能會導致性能問題和用戶體驗不佳,因為這些函數可能需要執行複雜的操作,如計算、網路請求等。 為了優化這種情況,我們可以使用防抖和節流來限制函數的調用次數,從而提高性能和用戶體驗。 防抖 防抖是指在 ...
  • 簡介 原型模式(Prototype Pattern)是一種創建型設計模式,使你能夠複製已有對象,而無需使代碼依賴它們所屬的類,同時又能保證性能。 這種模式是實現了一個原型介面,該介面用於創建當前對象的克隆。當直接創建對象的代價比較大時,則採用這種模式。 如果你需要複製一些對象,同時又希望代碼獨立於這 ...
  • 閱讀原文:https://bysocket.com/openai-chatgpt-vs-developer/ ChatGPT 能取代多少程式員的工作?導致我們程式員失業嗎?這是一個很好的話題,我這裡分享下: 一、ChatGPT 是什麼?有什麼作用 ChatGPT是一種基於人工智慧技術的語言模型,是可 ...
  • 環境:CentOS 7.6_x64Python版本:3.9.12FreeSWITCH版本 :1.10.9 一、背景描述 ESL庫是FreeSWITCH對外提供的介面,使用起來很方便,但該庫是基於C語言實現的,Python使用該庫的話需要使用源碼進行編譯。如果使用系統自帶的Python版本進行編譯,過 ...
  • 高級特性 主要內容 不安全 Rust 高級 Trait 高級 類型 高級函數和閉包 巨集 一、不安全 Rust 匹配命名變數 隱藏著第二個語言,它沒有強制記憶體安全保證:Unsafe Rust(不安全的 Rust) 和普通的 Rust 一樣,但提供了額外的“超能力” Unsafe Rust 存在的原因: ...
  • 這一年乾的很多事都是為了降低我的開源項目消息推送平臺austin使用門檻。 如果想學Java項目的,強烈推薦我的開源項目消息推送平臺Austin(8K stars) ,可以用作畢業設計,可以用作校招,可以看看生產環境是怎麼推送消息的。開源項目消息推送平臺austin倉庫地址: 消息推送平臺🔥推送下 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...