美文网首页Swift相关runtime相关Swift
Swift3.0 利用 Runtime 简单封装一个字典转模型

Swift3.0 利用 Runtime 简单封装一个字典转模型

作者: i_have_an_Apple | 来源:发表于2016-11-16 18:12 被阅读1211次

    在通常的项目中,我们经常会用到字典转 model 的操作,我们可以使用系统的
    setValuesForKeys(Swift)
    setValuesForKeysWithDictionary(OC)
    方法来完成这一操作,但是这样就会遇到一个问题,如果我们数据字典其中的一个 key 与系统关键字重名,那我们在model中使用这个 key 作为属性就会报错,为了解决这一问题,我们会使用一些第三方库去完成字典转模型的操作,例如 MJExtension ,在这里,我们自己去封装一个简单的字典转模型,闲话不多说,我们马上开始。

    首先我们去创建一个 BaseModel 类,我们在这个根类中去实现一个可以字典转自身属性的构造方法,只要我们自定义的 model 都继承这个 BaseModel 那么我们的 model 就都能使用这个构造方法完成字典转模型的操作啦~

    class BaseModel: NSObject { 
        //自定义构造方法
        init(dic: [String:Any]) {
            super.init()
        }
    }
    

    现在,我们已经通过构造方法,拿到了数据字典,那么接下来我们只要将字典的键值对转换为我们自身的属性,就大功告成啦~
    我们写一个新的方法,去完成这个操作
    我们首先在这个方法中使用 Runtime 获取一下本类的所有属性

    func setAttribut(dic: [String:Any]) -> Void {
            //Runtime获取本类属性
            var count:UInt32 = 0
            let ivars = class_copyIvarList(self.classForCoder, &count)
    }
    

    然后我们遍历这个获取到的属性数组,取出其中的元素,并获得属性名,这里值得注意的是,我们获得的属性名是 C 语言字符串,这里我们要转换一下变成 Swift 字符串

    for i in 0..<count {
             //取出属性名
             let ivar = ivars?[Int(i)]
             let ivarName = ivar_getName(ivar!)
             let nName = String(cString: ivarName!)
    }
    

    进行到这一步,相信很多小伙伴已经明白其中的原理了,接下来,我们只要利用取到的属性名从我们的数据字典中取到相应的 value 然后赋值给我们的属性,我们的任务就完成了,但是这里,我们要解决我们刚开始遇到问题 “我们的属性名和字典的key值必须不相同怎么办?” 在这里我的解决办法是重新建立一个 model 属性与字典 key 值的映射关系,这里又写了一个建立映射的方法

    //如果属性名与数据字典的key值不对应,那么在子类model中复写此方法,将属性名作为key,字典key值作为value
        func attributesDic(dic: [String:Any]) -> [String:String] {
            var newDic:[String:String] = [:]
            for key in dic.keys {
                //复写时注意将属性名作为key 数据字典的key作为value
                newDic[key] = key
            }
            return newDic
        }
    

    在这个 BaseModel 父类中,我们先让数据字典所有的 key 映射为 key 本身,这样我们在复写这个方法时只修改 key 与属性不对应的映射就可以了。
    这里有特别注意的一点,在复写时,我们一定要用 super 首先调用一下这个方法。

    这样,我们的属性赋值方法就要修改了,我们要首先拿到数据字典的 key 与属性的全新映射关系

    func setAttribut(dic: [String:Any]) -> Void {
            //获得映射关系
            let attributDic = attributesDic(dic: dic)        
            //Runtime获取本类属性
            var count:UInt32 = 0
            let ivars = class_copyIvarList(self.classForCoder, &count)
            for i in 0..<count {
                //取出属性名
                let ivar = ivars?[Int(i)]
                let ivarName = ivar_getName(ivar!)
                let nName = String(cString: ivarName!)
            }  
    }
    

    这样一来我们离成功就只差一步了!!
    我们需要将取到的属性名通过全新的映射关系取到数据字典的 key ,然后利用这个 key 从数据字典取到 value 最后将 value 赋值给我们 model 的属性
    最后,我们的属性赋值方法变成了这样

    func setAttribut(dic: [String:Any]) -> Void {
            let attributDic = attributesDic(dic: dic)
            //Runtime获取本类属性
            var count:UInt32 = 0
            let ivars = class_copyIvarList(self.classForCoder, &count)
            for i in 0..<count {
                //取出属性名
                let ivar = ivars?[Int(i)]
                let ivarName = ivar_getName(ivar!)
                let nName = String(cString: ivarName!)
                //取出要赋值的值
                var attribut = attributDic[nName]
                if attribut == nil{
                    attribut = ""
                }
                var value:NSObject
                if dic[attribut!] != nil {
                    value = dic[attribut!] as! NSObject
                } else {
                    value = "" as NSObject
                }
                //利用KVC给本类的属性赋值
                self.setValue(value, forKey: nName)   
            }        
    }
    
    最后的最后

    在我们自定义的初始化方法中调用一下

    //自定义构造方法
        init(dic: [String:Any]) {
            super.init()
            setAttribut(dic: dic)
        }
    

    大功告成!!
    这个封装好的 model 已经在我写的 Swift 小项目中得到了验证,这是项目地址
    时光电影Swift版初学小项目
    本文如果有什么错误或者您有更好的方法,欢迎指出

    相关文章

      网友评论

      本文标题:Swift3.0 利用 Runtime 简单封装一个字典转模型

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