很多时候我们需要把从服务器端请求下来的数据转成model类,今天就介绍一下如何利用runtime实现字典转模型

1、首先先建一个model类

class Person:NSObject { 
    var name:String?
    var age:NSNumber?
}

2、为NSObject创建一个extension,在其中实现字典转模型

主要分为一下几步

(1)获取所有的属性的名字

(2)通过属性的名字去字典里取值

(3)用KVC为model类赋值

extension NSObject{
    class func objectWithkeyvalues(keyvalues:NSDictionary) -> AnyObject{
        let model = self.init()
        //存放属性的个数
        var outCount:UInt32 = 0
        //获取所有的属性
        let properties = class_copyPropertyList(self.classForCoder(),&outCount)
        //遍历属性
        for var i = 0;i 

3、测试结果


但是有时候我们经常会遇到这样的情况model类继承自父类,比如

class Person:NSObject {
    var name:String?
    var age:NSNumber?
}
class Student:Person{
    var number:String?
    var score:NSNumber?
}
Student类继承自Person类我们也要为Student类中父类的属性赋值,也就是说我们在遍历属性的同时也要为父类的属性 ###4、遍历父类中得属性 在这种情况下我改变了一下思路,在获取属性的同时不在用KVC直接赋值,而是把获取的属性存起来,得到所有的属性之后在进行赋值 把扩展中得代码改造之后如下
extension NSObject{
    class func objectWithkeyvalues(keyvalues:NSDictionary) -> AnyObject{
        let model = self.init()
        //获取所有的属性
        let properties = self.allProperties()
        if let _ = properties{
            for property in properties!{
                if let value = keyvalues[property.propertyNmae]{
                    //为model类赋值
                    model.setValue(value,forKey: property.propertyNmae as String)
                }
            }
        }
        return model
    }
    class func allProperties() -> [LKKProperty]?{
        let className = Nsstring(CString: class_getName(self),encoding: NSUTF8StringEncoding)
            if let _ = Nsstring(CString: class_getName(self),encoding: NSUTF8StringEncoding){
                //不用为NSObject的属性赋值
                if className!.isEqualToString("NSObject"){
                    return nil
                }
            }else{
                return nil
            }
            var outCount:UInt32 = 0
            //所有属性LKKProperty里面放着存放这个属性
            var propertiesArray = [LKKProperty]()
            let properties = class_copyPropertyList(self.classForCoder(),&outCount)
            //获取父类的所有属性
            let superM = self.superclass()?.allProperties()
            if let _ = superM{
                //把父类中得所有属性添加进去
                propertiesArray += superM!
            }
            //遍历自己的属性添加进去
            for var i = 0;i 

5、我们还经常遇到这种情况,类型嵌套,比如Person类里面还有一个Card类代表着我们的身份信息,我们在赋值的时候也想直接把Card的所有属性都能直接弄好,也就是说我们在遍历属性

思路如下,在获取属性的过程中判断是否属于Foundtation框架,如果是直接赋值,如果不是就获取这个类的所有属性,对这个类的属性进行赋值,然后把Card这个类的对象赋值给Person这个类的对象,修改代码如下。

class LKKProperty{
    //属性名字
    var propertyNmae:Nsstring!
    //属性
    var property:objc_property_t
    //属性类型
    var propertyType:LKKType!
    init(property:objc_property_t){
        self.property = property
        self.propertyNmae = Nsstring(CString: property_getName(property),encoding: NSUTF8StringEncoding)
        //自定义的类的描述格式为T@"_TtC15字典转模型4Card",N,&,Vcard
        //T+@+"+..+工程的名字+数字+类名+"+,+其他,而我们想要的只是类名,所以要修改这个字符串
      //获取类的描述
        var code: Nsstring = Nsstring(CString: property_getAttributes(property),encoding: NSUTF8StringEncoding)!
        //直接取出""中间的内容
        code = code.componentsSeparatedByString("\"")[1]
        let bundlePath = getBundleName()
        let range = code.rangeOfString(bundlePath)
        if range.length > 0{
            //去掉工程名字之前的内容
            code = code.substringFromIndex(range.length + range.location)
        }
        //在去掉剩下的数字
        var number:String = ""
        for char in (code as String).characters{
            if char 

6、未完成的事情

(1)当类的属性与字典里的key值不一定的时候,出现的情况:字典里面的key是关键字的时候

(2)当类的属性是数组,并且数组里面要放自定义类的时候

接着完成未完成的事情,首先当字典里的key值与属性不一致的时候,我弄了个映射

一、解决类的属性与字典里的key值不一定的情况

//如果需要映射关系,就让子类复写此方法,获取映射到得值
//并且在LKKProperty这个类中添加key属性,用来取的字典Key
  func replacedKeyFromPropertyName() -> NSDictionary{
        return ["属性名字":"key名字"]
    }

二、首先我们添加一个方法,这个方法的作用是把字典数组转成模型数组。代码如下

//把一个字典数组转成一个模型数组
    class func objectArrayWithkeyvaluesArray(array:NSArray) -> [AnyObject]{
        var temp = Array<AnyObject>()
        let properties = self.allProperties()
        for(var i = 0;i < array.count;i++){
            let keyvalues = array[i] as? NSDictionary
            if (keyvalues != nil){
                let model = self.init()
                //为每个model赋值
                model.setValuesForProperties(properties,keyvalues: keyvalues!)
                temp.append(model)
            }
        }
        return temp
    }

三,当判断类的属性是数组的时候,通过这个方法拿到数组里面的类型

    //子类重写这个方法,说明数组里存放的对象类型
    func objectClassInArray() -> [String:String]{
        return ["属性名":"自定义类名"]
    }

四、在赋值时若发现是数组并且是数组里装的是自定义类的时候,用二的方法得到数组对象,并且赋值

    //把一个字典里的值赋给一个对象的值
    func setValuesForProperties(properties:[LKKProperty]?,keyvalues:NSDictionary){
        //判断属性数组是否存在
        if let _ = properties{
            for property in properties!{
                //判断该属性是否属于Foundtation框架
                if property.propertyType.isFromFoundtion {
                    if let value = keyvalues[property.key]{
                        //判断是否是数组,若是数组,判断数组里装的类是否是自定义类
                        if property.propertyType.isArray && property.propertyType.arrayClass != nil && value is NSArray{
                            //把字典数组转换成模型数组
                            let temp = property.propertyType.arrayClass!.objectArrayWithkeyvaluesArray(value as! NSArray)
                            //为model类赋值
                            self.setValue(temp,forKey: property.propertyNmae as String)
                        }else{
                            //为model类赋值
                            self.setValue(value,forKey: property.propertyNmae as String)
                        }
                    }
                }else{
                    if let value = keyvalues[property.key]{
                        if value is NSDictionary{
                            let subClass = property.propertyType.typeClass?.objectWithkeyvalues(value as! NSDictionary)
                            //为model类赋值
                            self.setValue(subClass,forKey: property.propertyNmae as String)
                        }
                    }
                }
            }
        }
    }

这些内容下篇文章见,demo下载地址

swift之用runtime实现字典转模型的更多相关文章

  1. ios – 声明NSDictionary并在Swift中添加键值对?

    我一直在尝试使用类类型键和值来声明一个NSDictionary,如下所示:这里,“Category”和“SubCategory”是全局类.我知道我不能将类类型用于关键字段.但是,无论如何,我应该做到这一点.有没有办法做到这一点?如何声明专门的NSDictionary或类似的东西来做到这一点?

  2. ios – Swift相当于`[NSDictionary initWithObjects:forKeys:]`

    Swift的原生字典是否与[NSDictionaryinitWithObjects:forKeys:]相当?假设我有两个带键和值的数组,并希望将它们放在字典中.在Objective-C中,我这样做:当然我可以通过两个数组迭代一个计数器,使用vardict:[String:Int]并逐步添加东西.但这似乎不是一个好的解决方案.使用zip和enumerate可能是同时迭代两者的更好方法.然而,这种方法

  3. ios – 在Swift中获取Cocoa Touch Framework项目版本字符串

    有谁知道这是否是我的项目设置中的缺陷,Xcode中的一个错误,或者是否有一种方法可以将Swift中的框架版本作为String或数组获取,这样我可以提供比major.minor更精细的版本控制?

  4. ios – 搜索数组swift中的对象

    我正在尝试使用UISearchController创建搜索功能.但是,我似乎无法使其与我的团队对象一起工作.我首先创建了一个包含id,name和shortname的TeamObject.然后我从一个url中检索teamData,并将TeamObjects添加到一个填充到tableView中的数组中.这个tableView包含一个searchController,它假设过滤数据,但没有任何反应.阵列

  5. ios – Swift可选项:语言问题,还是做错了什么?

    应该有可选的类型;type是但是,如果我这样做,它的工作原理:它似乎是基本的替代,但我可能会遗漏一些语言的细微差别.谁能对此有所了解?之后就像暧昧一样,更多,这是我的解决方案:这适用于所有非对象Swift对象,包括Swift字符串,数字等.感谢Viktor提醒我String不是Swift中的对象.如果您知道值的类型,您可以替换任何?使用适当的可选类型,如String?

  6. ios – Swift – 使用字典数组从字典访问数据时出错

    我有一个非常简单的例子,说明我想做什么基本上,我有一个字典,其值包含[String:String]字典数组.我把数据填入其中,但当我去访问数据时,我收到此错误:Cannotsubscriptavalueoftype‘[([String:String])]?’withanindexoftype‘Int’请让我知道我做错了什么.解决方法您的常量数组是可选的.订阅字典总是返回一个可选项.你必须打开它.更

  7. ios – 在Swift中使用“Map”创建两个数组的超集

    假设我有两个数组:我想组合两个数组,以便我得到一个输出我该怎么做呢?

  8. 模糊地使用’下标’ – ios 9 Swift 2.0

    我在xcode7.0上使用Swift2.0编写iosapp.在更新到最新版本的xCode7.1之前,完全相同的代码完全正常更新后,我收到此错误:Ambiguoususeof‘subscript’在这些方面:这是全班:Theoriginallibrary解决方法编译器不知道self.itemAttributes[indexPath.section]返回的内容,因为它定义为NSMutableArray

  9. xcode – Swift – 检索子视图

    解决方法UIViewController没有子视图属性.它有一个view属性,它有一个subviews属性:但通常这不是一个好主意.您应该将所需的标签放入IBOutletCollection并迭代它.否则,您与确切的子视图集密切相关.要创建IBOutletCollection,请在IB中选择所需的所有标签,然后将其控制拖动到源代码中.它应该询问您是否要创建一个集合数组.

  10. iOS Swift – 如何使用Core Data存储数组?

    我是iOS开发的新手,想要知道我应该指定哪种数据类型来存储多个字符串(数组).该应用程序与食物有关,我需要将多种成分存储为一个属性.我当时正在考虑将原料作为实体,但我只是想让原料变得容易.我已阅读有关可转换类型但人们似乎并不建议使用它来存储数组.解决方法警告:提前见解答.你没有.将数据存储在数组中并不会使您更容易.相反,它会让事情变得更加困难一小时.想象一下,你想要显示包含所选成分的所有食谱.对于

随机推荐

  1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

    下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

  2. swift UISlider,UIStepper

    我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

  3. preferredFontForTextStyle字体设置之更改

    即:

  4. Swift没有异常处理,遇到功能性错误怎么办?

    本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

  5. 字典实战和UIKit初探

    ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

  6. swift语言IOS8开发战记21 Core Data2

    上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

  7. swift语言IOS8开发战记22 Core Data3

    上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

  8. swift实战小程序1天气预报

    在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

  9. 【iOS学习01】swift ? and !  的学习

    如果不初始化就会报错。

  10. swift语言IOS8开发战记23 Core Data4

    接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

返回
顶部