美文网首页SwiftSwift
Swift4 json Decodable 解析指南

Swift4 json Decodable 解析指南

作者: 小奉不在乎 | 来源:发表于2017-10-18 14:45 被阅读615次

    没有多好的文笔,也没有多大的智慧,网上搜索Decodable资料有多少我就不知道了,我只想要傻瓜式的,一句话调用,json转模型!

    重要的事情说三遍(Decodable是苹果官方的 Swift4新特性Decodable是苹果官方的 Swift4新特性Decodable是苹果官方的 Swift4新特性

    假设你的json长成这样

    {
      status:1000,
      message:“操作成功”,
      data:{
        test:"我是测试数据"
      }
    }
    

    或者这样

    {
      status:1000,
      message:“操作成功”,
      data:{
        Test:{
          detail:"测试内的数据",
          detailMessage:“我也是测试内的数据,
        }
      }
    }
    

    怎么办呢,别急,我们一步一步的来,虽然看起来写比较粗糙,实用就好,当然你有更好的写法记得告诉我一下,技术因分享而有价值!

    第一步 新建DecodableModels.Swift
    import Foundation
    
    /// 只有状态值和提示信息的模型转化
    struct ModelObjet: Decodable {
        /// 状态值
        var status:Int? = nil
        /// 提示信息
        var message:String? = nil
    }
    
    /// 转换成数组模型
    struct ModelObjetT<T: Decodable>: Decodable {
        /// 状态值
        var status:Int? = nil
        /// 提示信息
        var message:String? = nil
        /// 嵌套模型
        var data:T? = nil
    }
    
    /// 转换成数组模型
    struct ModelArrayT<T: Decodable>: Decodable {
        /// 状态值
        var status:Int? = nil
        /// 提示信息
        var message:String? = nil
        /// 嵌套模型
        var data:[T]? = nil
    }
    

    注: var status:Int? = nil 这种写法是因为Swift的类型推断会很慢,这样手动指定,会提高编译器运行效率!

    第二步 写上两个分类
    extension Data {
        /// json data 转模型
        /// 可以直接拿网络请求成功的response直接转换
        func jsonDataMapModel<T: Decodable>(_ type: T.Type) -> T? {
            let decoder = JSONDecoder()
            do {
                return try decoder.decode(T.self, from: self)
            } catch {
                print("解析失败\(self)")
                return nil
            }
        }
    }
    
    extension String {
        /// json字符串转模型
        func jsonStringMapModel<T: Decodable>(_ type: T.Type) -> T? {
            
            if let jsonData = self.data(using: .utf8) {
                return jsonData.jsonDataMapModel(T.self)
            }
            print("解析失败\(self)")
            return nil
        }
    }
    
    第三步 搞事情

    模拟创建一个json字符串

    let jsonString = "{\"status\":1000,\"message\":\"操作成功\",\"data\":{\"test\":\"测试文字\"}}"
    

    然后
    创建一个Test模型

    struct Test:Decodable {
    
        var test:String?
    }
    

    我们就可以输出

    let model = jsonString.jsonStringMapModel(ModelObjetT<Test>.self)
            print(model?.message ?? "")
    

    或者是这样

    print(model?.data?.test ?? "")
    

    开发过程中发现一个didSet无效的问题

    class Test:Decodable {
        var name: String?
        /// 测试文字
        var test:String?{
            didSet{
                name = test
            }
        }
    }
    // json字符串一键转模型
    let jsonString = "{\"status\":1000,\"message\":\"操作成功\",\"data\":{\"test\":\"0.05\"}}"
    let model = jsonString.jsonStringMapModel(ModelObjetT<Test>.self)
    print(model?.data?.test ?? "test无值")
    print(model?.data?.name ?? 0.00)
    

    无情的打印结果

    0.05
    0.00
    

    更改如下

    class Test:Decodable {
        lazy var name: Double? = { return Double(test ?? "0.00") ?? 0.00 * 100 }()
        /// 测试文字
        var test:String?
    }
    

    这下就杠杠的了

    0.05
    5.0
    

    来波完整的,新建一个.playground粘贴进去即可

    import UIKit
    
    /// 只有状态值和提示信息的模型转化
    struct ModelObjet: Decodable {
        /// 状态值
        var status:Int? = nil
        /// 提示信息
        var message:String? = nil
    }
    
    /// 转换成数组模型
    struct ModelObjetT<T: Decodable>: Decodable {
        /// 状态值
        var status:Int? = nil
        /// 提示信息
        var message:String? = nil
        /// 嵌套模型
        var data:T? = nil
    }
    
    /// 转换成数组模型
    struct ModelArrayT<T: Decodable>: Decodable {
        /// 状态值
        var status:Int? = nil
        /// 提示信息
        var message:String? = nil
        /// 嵌套模型
        var data:[T]? = nil
    }
    
    extension Data {
        /// json data 转模型
        /// 可以直接拿网络请求成功的response直接转换
        func jsonDataMapModel<T: Decodable>(_ type: T.Type) -> T? {
            let decoder = JSONDecoder()
            do {
                return try decoder.decode(T.self, from: self)
            } catch {
                print("解析失败\(self)")
                return nil
            }
        }
    }
    
    extension String {
        /// json字符串转模型
        func jsonStringMapModel<T: Decodable>(_ type: T.Type) -> T? {
            
            if let jsonData = self.data(using: .utf8) {
                return jsonData.jsonDataMapModel(T.self)
            }
            print("解析失败\(self)")
            return nil
        }
    }
    
    class Test1:Decodable {
        lazy var name: Double? = { return (Double(test ?? "0.00") ?? 0.00) * 100 }()
        /// 测试文字
        var test:String?
    }
    class Test2:Decodable {
        lazy var name: String? = { return "我是test_name转换之后的\(test_name ?? "")" }()
        /// 测试文字
        var test_name:String?
        
        var detial:[Detial]?
        
    }
    class Detial:Decodable {
        var detial_name: String?
    }
    
    
    // json字符串一键转模型
    func test1(){
        let jsonString = "{\"status\":1000,\"message\":\"操作成功\",\"data\":{\"test\":\"0.05\"}}"
        let model = jsonString.jsonStringMapModel(ModelObjetT<Test1>.self)
        print(model?.data?.test ?? "test无值")
        print(model?.data?.name ?? 0.00)
        print("============华丽的分割线==============")
    }
    func test2(){
        let jsonString = "{\"status\":1000,\"message\":\"操作成功\",\"data\":{\"test_name\":\"Decodable\",\"detial\":[{\"detial_name\":\"看吧嵌套毫无压力\"}]}}"
        let model = jsonString.jsonStringMapModel(ModelObjetT<Test2>.self)
        print(model?.data?.test_name ?? "test无值")
        print(model?.data?.name ?? "name无值")
        print(model?.data?.detial?.first?.detial_name ?? "detial_name无值")
    }
    
    test1()
    test2()
    

    好啦,今天就先扯到这,有什么不懂的可以加QQ群384089763我们一起探讨,咱们不见不散!

    相关文章

      网友评论

      • Be_ingenious:这样数据类型不对时就会解码失败,有什么好的解决方法么
        小奉不在乎:@Be_ingenious https://github.com/Flight-School/AnyCodable这个可能对你有帮助,我这个只是入门级的写法,供学习使用而已,具体使用要根据项目来进行一个封装。

      本文标题:Swift4 json Decodable 解析指南

      本文链接:https://www.haomeiwen.com/subject/jgteuxtx.html