swift 字典转模型框架

import Foundation

@objc protocol DictModelProtocol{

///自定义映射
///
/// :returns: 返回可选映射关系字典 [属性名: 自定义对象名称]
/// :class 替换成 static 是 swift 1.2 修改的
static func customClassMapping() -> [String: String]?

}
///字典转模型工具
class SwiftDictModel {

//单例,全局访问入口

static let sharedManger = SwiftDictModel()



///  将字典转换成模型对象
///
///  :param: dict 数据字典
///  :param: cls  模型类
///
///  :returns: 实例化的类对象
func objectWithDictionary(dict:NSDictionary,cls:AnyClass) -> AnyObject?{

    //1. 取出模型类的字典
    let dictInfo = fullModelinfo(cls)
    
    //实例化对象
    var obj:AnyObject = cls.alloc()
    
    //遍历模型字典,有什么属性就设置什么属性
    // k 应该 和 dict 中的 key 是一致的
    for (k,v) in dictInfo {
  
        //取出字典中的内容
        if let value: AnyObject? = dict[k]{
            
            println("要设置数值的 \(value) + key \(k)")
            //判断是否是自定义类
            //json 反序列化的时候,如果是null 值,保存在字典中的事NSNull()
            if v.isEmpty && !(value === NSNull()){
               obj.setValue(value, forKey: k)
                
            }else{
                
                let type = "\(value!.classForCoder)"
                     println("\t自定义对象 \(value) \(k) \(v) ---- type \(type)")
                
                //两种可能,字典/数组
                if type == "NSDictionary"{
                    // value 是字典-> 将 value 的字典转换成 Info 的对象
                    if let subObj: AnyObject? = objectWithDictionary(value as! NSDictionary, cls: NSClassFromString(v)){
                    
                        //使用kvc设置数值
                        obj.setValue(subObj, forKey: k)
                        
                    }
                    
                    }else if type == "NSArray" {
                    
                        //value 是数组
                        //如果是数组如何处理? 遍历数组,继续处理数组中的字典
                        if let subObj:AnyObject? = objectWithArray(value as! NSArray, cls: NSClassFromString(v)){
                        obj.setValue(subObj, forKey: k)
                        }
                    
                    }
                
                }
                
            }
            
        
        }
    

    println(dictInfo)
    
    return obj

}

///将数组转换成模型字典
///
/// :parm: array 数组的描述
/// : param: cls 模型类
///
/// :returns: 模型数组

func objectWithArray(array: NSArray, cls: AnyClass) -> [AnyObject]? {

    //创建一个数组
    var result = [AnyObject]()
    
    //1.遍历数组
    //可能存在什么类型? 字典/数组
    for value in array{
    
        let type = "\(value.classForCoder)"
        
        if  type == "NSDictionary"{
        
            if let subObj:AnyObject = objectWithDictionary(value as! NSDictionary, cls: cls){
            
                result.append(subObj)
            
            }
        }else if type == "NSArray"{
         
            if let subObj: AnyObject = objectWithArray(value as! NSArray, cls: cls){
            
                result.append(subObj)
            }
            
        }
    
    }
    
return result

}

/// 缓存字典 格式[类名:模型字典, 类名2:模型字典]
var modelCache = [String:[String:String]]()

/// 获取模型类的完整信息
/// 
///: parm:cls  模型类
func fullModelinfo(cls:AnyClass)-> [String: String] {

    // 判断类信息是否已经被缓存
    if let cache = modelCache["\(cls)"] {
        println("\(cls)已经被缓存")
        return cache
    }
    var currentCls: AnyClass = cls
    
    //模型字典
    var dictInfo = [String:String]()
    
    while let parent:AnyClass = currentCls.superclass(){
    
        //取出并且拼接 currentCls 的模型字典
        dictInfo.merge(modelInfo(currentCls))
        
        currentCls = parent
        println(parent)
    
    }
    // 将模型信息写入缓存
   modelCache["\(cls)"] = dictInfo
    
    println(dictInfo)
   return dictInfo

}
//获取给定类的信息
func modelInfo(cls:AnyClass) -> [String: String]{

    // 判断类信息是否已经被缓存
    if let cache = modelCache["\(cls)"] {
        println("\(cls)已经被缓存")
        return cache
    }
    
    var mapping: [String:String]?
    if cls.respondsToSelector("customClassMapping"){
    
        println("实现了协议")
        
        //调用协议方法,获取自定义对象映射关系字典
        
        mapping = cls.customClassMapping()
        
        println(mapping)
    }
    //获取累的属性
    var count: UInt32 = 0
    
    let ivars = class_copyIvarList(cls, &count)
    
    println("有\(count)属性")

// // 定义一个类属性的字典:[属性的名字,自定对象的名称/“”]

    var dictInfo = [String: String]()
    for i in 0..<count {
    
        //检索数组下标只能,用Int 
        let ivar = ivars[Int(i)]
        
        //UInt8 = char, c语言的字符串
        let cname = ivar_getName(ivar)
        //将c 语言字符串转换成 swift 的String
        let name = String.fromCString(cname)!
         //println(name)
        let type = mapping?[name] ?? ""
        
        //设置字典
        dictInfo[name] = type
        
    
    }
    
    free(ivars)
    
    // 将模型信息写入缓存
    modelCache["\(cls)"] = dictInfo
    return dictInfo
}

}

extension Dictionary{

mutating func merge<K, V>(dict:[K:V]){

    for (k,v) in dict{
    
        //字典的分类法, 如果要使用 updateValue 需要明确的指定类型
        self.updateValue(v as! Value, forKey: k as! Key)
    
    }

}

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,698评论 0 9
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,080评论 4 62
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 1. genuine adj. 真的,真正的;真诚的 2. emergency n. 紧急情况,突发事件 ...
    单单单单单阅读 316评论 2 2
  • 精力汤食材:(3人份) 生菜4片 小番茄3个 芒果1个 地瓜4片 银耳1朵(熟)黄豆60克(熟)枸杞10克 葡萄干...
    素之味阅读 353评论 0 0