一、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.quest)
四、问题实战
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