swift codable

一、codable功能

OC常用模式:

 

 

 示例代码如下:

NSDictionary *metaDic =
        [NSJSONSerialization JSONObjectWithData:metaData options:0 error:&error];

- (instancetype)initWithContentDictionaty:(NSDictionary *)contentDic {
    self = [super init];
    if (self) {
        //_blockId = [[contentDic valueForKey:@"block_id"] integerValue]; //V4去除
        _blockType = [[contentDic valueForKey:@"block_type"] integerValue];
        _blockIndex = [[contentDic valueForKey:@"block_index"] integerValue];
        _blockStyle = [contentDic valueForKey:@"block_style"];
        _blockContent = [contentDic valueForKey:@"block_content"];
        _imgUUID = [contentDic valueForKey:@"img_uuid"];
        //_storyId = [[contentDic valueForKey:@"story_id"] integerValue]; //v4去除
    }
    
    return self;
}

swift codable实现从json字符串 直接转 model

二、原理

codable协议分为 解码codable和编码codable

typealias Codable = Decodable & Encodable

协议实现

protocol Encodable {
    func encode(to encoder: Encoder) throws
}

protocol Decodable {
    init(from decoder: Decoder) throws
}

使用方法

class Person:Codable {
    var name:String
    var age:Int 
    var quest:String
}

 let dic = ["name":"huahua","age":10,"quest":"1"] as [String : Any]
        let JSONString = "{\"name\":\"xiaoming\",\"age\":10,\"quest\":\"xiaoming\"}"
        guard let jsonData = JSONString.data(using: .utf8)else{
           return  
        } 
        
        let decoder = JSONDecoder()
        guard let obj  = try?decoder.decode(Person.self, from: jsonData)else{
            return
        }
        print(obj.name)
        print(obj.age)
        print(obj四、问题实战

1、json.key > model.property, 成功解析
2、json.key < model.property,解析失败,只要class中又一个属性在json.key中找不到,解析失败
解决方案一:sever有些字段可传可不传,将model中多出来的字段 设置为可选
解决方案二:sever有些字段永远不传入,重写codingKeys
3、sever字段与前端 关键字冲突解决方案
前端字段加上'default'
4、对象引用的枚举类型数量 和 sever不匹配
重写枚举类型初始化方法

enum RequstState:String,Codable,RawRepresentable{
    case success = "success"
    case failed = "failed"
    case unkown
}

extension RequstState{
    //可失败的初始化构造器
    init?(rawValue: Self.RawValue){
        switch rawValue {
        case "failed":
            self = .success
        case "success":
            self = .failed
        default:
            self = .unkown
        }
    }
}
5、何时需要写codingKey
前端定制字段名称
6、何时重写decoder
前端定制decode方案,默认值赋值等
7、何时重写encoder
前端定制enCode方案,默认值赋值等,如下为重写decode示例代码
struct Student:Codable{
  var id:String
//  var name:String
  var grade:Int?
  var state:RequstState
    
  init(from decoder: Decoder) throws{
      let container = try decoder.container(keyedBy: CodingKeys.self)
      do {
               state = try container.decodeIfPresent(RequstState.self, forKey: .state)
           } catch {
               state = nil
           }
    do {
        grade = try container.decodeIfPresent(Int.self, forKey: .grade)
    } catch {
        grade = nil
    }
    do {
        id = try container.decodeIfPresent(String.self, forKey: .id) ?? ""
    } catch {
        id = ""
    }
  }
}
五、项目遇到特殊问题

1、codable json串中 value为string ,string包含 转义字符的 json 字典格式串,这种形式只能先解包成string,然后特殊处理

 

 

code中的样子大致如下图

 注:"""三引号表示 中间的字符串 保留换行格式

六、扩展

1、string Dictionay 相互转化

 

 2、model to string

var snsDescription: String {
        
        guard let data  = try? JSONEncoder().encode(self)else{
            return "\(type(of: self)) 模型转 json字符串失败"
        }
        
        guard let str = String.init(data: data, encoding: .utf8) else {
            return "\(type(of: self)) 模型转 json字符串失败"
        }
        
        return str
    }
理论上讲 请求发送 也可以封装成 model 传入 请求模型,请求库负责转成string 或 dic 发送网络请求,目前实际使用中,使用的是Dic ,code如下图,组装字典过程中,有使用到model to string 的例子,UgcDynamicPic ---> Dic

参考:https://www.jianshu.com/p/36eccbacfb0a

posted @ 2021-01-08 14:40  心泪无恒  阅读(816)  评论(0编辑  收藏  举报