【小弟我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记10 Property List

【我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记10 Property List

  这一话来讲一个AnyObject的应用:Property List。

【小弟我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记10 Property List

property list不是任何一种类型,它属于一种工具类的东西。它表面上是AnyObject,但是在后台,它是NSString、NSArray、NSDictionary、NSNumer、NSData和NSDate这六种类型中的一种,或者它们桥接到Swift中的版本。我们为什么要用property list呢,它看起来很笨重,我们需要大量的类型转换。

实际上有两个原因,第一property list要用来隐式地传递数据,就像浏览器中的cookie一样,它们只有创建者才知道如何解释它们的一组数据,只有创建者才知道里面具体是Int还是String之类的,其他人能看到的只有AnyObject,所以他们嫩过的就是把它传递给其他人。

另一个原因是property list可以用于泛型的数据结构,泛型的数据结构可以用来写入磁盘,或者通过网络传播。

property list在IOS中有一个很好用的东西叫做NSUserDefaults(我在UI专项训练这个系列中使用了NSUserDefaults来记录用户登录情况,从而判断是否加载引导页)。NSUserDefaults就像一个微型的数据库,它只会存储property list,存储那些诸如设置、用户信息的东西。由于它的性能不高,所以千万不要用它存储一些图片什么的比较大的数据,只用它来存储property list,比如日期、字符串、数字什么的。它可以像字典一样存储和导出property list,而且它本身就是一个字典。下面是它的一些方法。

【小弟我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记10 Property List

它有一个key:value的字典,但是只存储property list。当你的APP关闭时,NSUserDefaults存储的东西依然存在,它永久保存这些数据。一般来说一个字典保存在堆中,当你的APP关掉它就消失了,但是这个保留着,所以说它像个数据库。

那么我们该如何使用它呢?

【小弟我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记10 Property List

首先你使用standarUserDefaults这个方法,给你一个NSUserDefaults的实例。你总是使用同一个实例,然后你就可以给它发送消息,使用objectForKey来读取一个信息或者使用setObject来写一个信息。通常数据会自动保存,但是也有强制保存数据的办法aunchronize,类似于save方法。那么既然有自动保存为什么还需要强制保存呢,这是因为保存机制是当你的App从前台转到后台了,那时它会保存数据。因此它不会总在我们需要的时候保存数据,这就需要我们强制保存数据,因此这种方法是需要的。特别是当你debugging的时候,中断操作不会保存数据,我们需要做一些保存。何况synchronize操作代价并不高。

现在来展示一个Demo

【小弟我们都爱Paul Hegarty】斯坦福IOS8公开课个人笔记10 Property List

在我们的计算器项目中,希望我们的CalculatorBrain能够给我们提供一个program,我们与这个program交互的数据都是AnyObject类型的,这样别人就看不到它到底是什么。

回到我们之前的Calculator项目中,在CalculatorBrain中添加如下代码:

    var program:AnyObject{//确保它是一个propertylist
        get{
        var returnValue = Array<String>()
            for op in opStack{
            returnValue.append(op.description)
            }
            return returnValue
        }
        set{
        
        
        }
    
    
    }

我们知道Array是一个AnyObject类型的,因为它被桥接过了。但是get中的代码显得有点多,之前介绍过一个叫做map的函数,正好派上用场,简化后的写法:

get{
            return opStack.map{$0.description}
        }

map后面是一个闭包,在闭包中把opStack中的每一个单个元素的字符描述返回到一个新的数组中。现在有人给我们返回了一个program,我们想要在其中加上我们的opStack,首先我们要确认的是别人给我们的这个数组中的数据类型。

var program:AnyObject{//确保它是一个propertylist
        get{
            return opStack.map{$0.description}
        }
        set{
            if let opSymbols = newValue as? Array<String>{
        var newOpStack = [Op]()
                for opSymbol in opSymbols{
                    if let op = knownOps[opSymbol]{
                    newOpStack.append(op)
                    } else if let operand = NSNumberFormatter().numberFromString(opSymbol)?.doubleValue {
                    newOpStack.append(.Operand(operand))
                    }
                
               opStack = newOpStack }
            }
        }
    
    
    }

set中的代码不打算细讲,如果你对计算器这个项目很熟悉的话应该是很好理解的。另外我们可以使用一种新的命名方式,那就是替身:

typealias PropertyList = AnyObject
    var program:PropertyList{//确保它是一个propertylist

这样更鲜明一点。