Codable?SwiftyJSON數(shù)據(jù)轉(zhuǎn)模型

轉(zhuǎn)模型步驟

- 1.獲取到數(shù)據(jù)后使用JSON進(jìn)行包裝
- 2.通過JSON獲取rawData
- 3.使用`JSONDecoder().decode<T>(_ type: T.Type, from data: Data)`轉(zhuǎn)為模型

decode函數(shù)

public extension JSON {
    func decodeObjcet<T>(_ type: T.Type) -> T? where T: Decodable {
        do {
            let data = try self.rawData()
            let result = try JSONDecoder().decode(type, from: data)
            return result
        } catch SwiftyJSONError.invalidJSON {
            if type == JSON.self {
                return self as? T
            }
            return self.rawValue as? T
        } catch {
            return nil
        }
    }
}

聲明數(shù)據(jù)模型

struct惠猿、class、enum可按需使用望艺,并遵守Codable協(xié)議。
舉例說明:
一個(gè)描述一個(gè)人姓名肌访、性別找默、教育經(jīng)歷等信息的json數(shù)據(jù)如下

{
    "age": 20,
    "name": "大碗",
    "sex": 1,
    "haveJob": false,
    "educations": [
          {
              "rank" : 1,
              "schoolName" : "春我部小學(xué)"
          },
          {
              "rank" : 2,
              "schoolName" : "春我部初級(jí)中學(xué)"
          },
          {
              "rank" : 3,
              "schoolName" : "春我部高級(jí)中學(xué)"
          }
    ]
}

可聲明數(shù)據(jù)模型如下

enum Sex: Int, Codable {
    case man = 1
    case woman = 2
}
enum EducationRank: Int, Codable {
    case primary = 1
    case juniorMiddle = 2
    case seniorMiddle = 3
    case university = 4
    case masterDegreeCandidate = 5
    case doctoralCandidate = 6
}
struct Education: Codable {
    var rank: EducationRank?
    var schoolName: String?
}
struct Person: Codable {
    /// 年齡
    var age: Int?
    /// 姓名
    var name: String?
    /// 性別
    var sex: Sex?
    /// 是否有工作
    var haveJob: Bool?
    /// 受教育經(jīng)歷
    var educations: [Education]?
}

調(diào)用var dawan = JSON(json).decodeObjcet(Person.self)即可完成最基礎(chǔ)的數(shù)據(jù)轉(zhuǎn)模型了。

數(shù)據(jù)類型Bool

上面的例子吼驶,網(wǎng)絡(luò)數(shù)據(jù)里惩激,表示Bool類型可能不會(huì)用true false,而是用10表示蟹演,那么使用Bool類型聲明的haveJob屬性則無法正常解析风钻,對(duì)于這種情況可以使用一個(gè)結(jié)構(gòu)體包裹數(shù)據(jù),并實(shí)現(xiàn)init(from decoder: Decoder)函數(shù)酒请,然后提供獲取bool值的方法或?qū)傩月饧肌wiftyJSON的 JSON類型正好可以滿足這個(gè)需求,聲明var haveJob: JSON?羞反,使用時(shí)布朦,調(diào)用JSONboolboolValue進(jìn)行判斷。

默認(rèn)值

網(wǎng)絡(luò)數(shù)據(jù)里昼窗,所有的字段都可能是缺失的是趴,故聲明的屬性都是可選擇類型。
但這樣做在寫代碼會(huì)產(chǎn)生大量的可選值判斷

if (dawan.age ?? 0) > 18 {
    // 成年
}
if let name = dawan.name, !name.isEmpty {
    // 名字
}
if dawan.haveJob ?? false {
    // 有工作
}

以上這些代碼澄惊,其實(shí)是相當(dāng)于給到這些屬性一個(gè)默認(rèn)值的唆途,即0、 ""缤削、 false窘哈。
為了給這些屬性設(shè)定默認(rèn)值,要使用到@propertyWrapper屬性包裝器:

/// 默認(rèn)值協(xié)議
public protocol DefaultValue {
    associatedtype Value: Codable
    static var defaultValue: Value { get }
}

@propertyWrapper
/// 默認(rèn)值包裝器
public struct Default<T: DefaultValue> {
    public var wrappedValue: T.Value
    public init(wrappedValue: T.Value) {
        self.wrappedValue = wrappedValue
    }
}
extension Default: Codable {
    /**
     ## 對(duì)Codable協(xié)議進(jìn)行處理
     ### decode時(shí)當(dāng)沒有對(duì)應(yīng)值時(shí)亭敢,賦值為默認(rèn)值
     ### encode時(shí)保證正確的結(jié)構(gòu)層級(jí)滚婉,避免 @propertyWrapper 對(duì)結(jié)構(gòu)層級(jí)造成`"key":{"wrappedValue"=value}`這樣的影響
     */
    public init(from decoder: Decoder) throws {
        let container = try decoder.singleValueContainer()
        wrappedValue = (try? container.decode(T.Value.self)) ?? T.defaultValue
    }
    public func encode(to encoder: Encoder) throws {
        var container = encoder.singleValueContainer()
        try container.encode(self.wrappedValue)
    }
}
public extension KeyedDecodingContainer {
    /// 當(dāng)json中不含有對(duì)應(yīng)key時(shí),為其賦值為默認(rèn)值
    func decode<T>(_ type: Default<T>.Type, forKey key: Key) throws -> Default<T> where T: DefaultValue {
        (try decodeIfPresent(type, forKey: key)) ?? Default(wrappedValue: T.defaultValue)
    }
}

然后給常用的幾種數(shù)據(jù)類型遵守DefaultValue協(xié)議

extension Bool: DefaultValue {
    public static var defaultValue: Bool = false
}
extension Int: DefaultValue {
    public static var defaultValue: Int = 0
}
extension Double: DefaultValue {
    public static var defaultValue: Double = 0
}
extension String: DefaultValue {
    public static var defaultValue: String = ""
}

枚舉類型默認(rèn)值

public protocol EnumCodable: RawRepresentable, Codable where RawValue: Codable {
    static var defaultCase: Self { get }
}
extension EnumCodable {
    public init(from decoder: Decoder) throws {
        if let container = try? decoder.singleValueContainer(), !container.decodeNil() {
            let decoded = try container.decode(RawValue.self)
            self = Self.init(rawValue: decoded) ?? .defaultCase
        } else {
            self = .defaultCase
        }
    }
}

對(duì)上面的Sex枚舉可以做如下改動(dòng)

enum Sex: Int, Codable {
    case unknow = 0
    case man = 1
    case woman = 2
}
extension Sex: EnumCodable {
    static var defaultCase: Sex = .unknow
}
extension Sex: DefaultValue {
    static var defaultValue: Sex = .defaultCase
}

也可以免去EnumCodable這一步

enum Sex: Int, Codable {
    case unknow = 0
    case man = 1
    case woman = 2
}
extension Sex: DefaultValue {
    static var defaultValue: Sex = . unknow
}

設(shè)定默認(rèn)值后的Person模型如下:

struct Person: Codable {
    /// 年齡
    @Default<Int>var age: Int
    /// 姓名
    @Default<String>var name: String
    /// 性別
    @Default<Sex>var sex: Sex
    /// 是否有工作
    var haveJob: JSON?
    /// 受教育經(jīng)歷
    var educations: [Education]?
}

數(shù)組默認(rèn)值

如果要對(duì)數(shù)組設(shè)定默認(rèn)值帅刀,可以使用typealias后遵守DefaultValue協(xié)議

typealias Educations = [Education]
extension Educations: DefaultValue {
    static var defaultValue: [Education] = [Education]()
}

@Default<Educations>var educations: Educations
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末让腹,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子扣溺,更是在濱河造成了極大的恐慌骇窍,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,734評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件锥余,死亡現(xiàn)場(chǎng)離奇詭異腹纳,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門嘲恍,熙熙樓的掌柜王于貴愁眉苦臉地迎上來足画,“玉大人,你說我怎么就攤上這事佃牛⊙痛牵” “怎么了?”我有些...
    開封第一講書人閱讀 164,133評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵俘侠,是天一觀的道長象缀。 經(jīng)常有香客問我,道長爷速,這世上最難降的妖魔是什么央星? 我笑而不...
    開封第一講書人閱讀 58,532評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮遍希,結(jié)果婚禮上等曼,老公的妹妹穿的比我還像新娘。我一直安慰自己凿蒜,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,585評(píng)論 6 392
  • 文/花漫 我一把揭開白布胁黑。 她就那樣靜靜地躺著废封,像睡著了一般。 火紅的嫁衣襯著肌膚如雪丧蘸。 梳的紋絲不亂的頭發(fā)上漂洋,一...
    開封第一講書人閱讀 51,462評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音力喷,去河邊找鬼刽漂。 笑死,一個(gè)胖子當(dāng)著我的面吹牛弟孟,可吹牛的內(nèi)容都是我干的贝咙。 我是一名探鬼主播,決...
    沈念sama閱讀 40,262評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼拂募,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼庭猩!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起陈症,我...
    開封第一講書人閱讀 39,153評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤蔼水,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后录肯,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體趴腋,經(jīng)...
    沈念sama閱讀 45,587評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,792評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了优炬。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片颁井。...
    茶點(diǎn)故事閱讀 39,919評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖穿剖,靈堂內(nèi)的尸體忽然破棺而出蚤蔓,到底是詐尸還是另有隱情,我是刑警寧澤糊余,帶...
    沈念sama閱讀 35,635評(píng)論 5 345
  • 正文 年R本政府宣布秀又,位于F島的核電站,受9級(jí)特大地震影響贬芥,放射性物質(zhì)發(fā)生泄漏吐辙。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,237評(píng)論 3 329
  • 文/蒙蒙 一蘸劈、第九天 我趴在偏房一處隱蔽的房頂上張望昏苏。 院中可真熱鬧,春花似錦威沫、人聲如沸贤惯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽孵构。三九已至,卻和暖如春烟很,著一層夾襖步出監(jiān)牢的瞬間颈墅,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評(píng)論 1 269
  • 我被黑心中介騙來泰國打工雾袱, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留恤筛,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,048評(píng)論 3 370
  • 正文 我出身青樓芹橡,卻偏偏與公主長得像毒坛,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子僻族,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,864評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容