Swift中的结构体与NSCoding

更新时间:2015-09-15 10:19:15点击次数:856次

正如大家所知,Swift中的结构体不遵守NSCoding协议。NSCoding只适用于继承自NSObject的类。可是结构体在Swift中的地位与使用频率都非常高,因此,我们需要一个能将结构体的实例归档和解档的方法。

Janie写过在Sonoplot工作时,他们团队对此的解决方法

简而言之,他们定义了一个拥有两个方法的协议:一个方法可以从结构体当中获得一个NSDictionary,另一个方法可以使用NSDictionary来初始化一个结构体。接着,再使用NSKeyedArchiver对这个NSDictionary进行序列化。这个方案的优雅之处在于,只要遵守了这个协议的结构体都可以进行序列化。

我最近灵光一闪,想到了另一种解决方案。尽管我已经实现了这种方案,并且使用它开发过几个小项目,但是我还是不确定这是不是一个好的方案。这个方法的优雅程度无法与上面提到的方法相提并论。然而我还是将它写出来,让读者自己来进行判断。

假设我们有一个person结构体:


[cpp] view plaincopy

  1. struct Person {  

  2.   let firstName: String  

  3.   let lastName: String  

  4. }  


我们不能使这个结构体遵守NSCoding协议,但是我们可以在结构体当中增加一个类的定义,使这个类来遵守NSCoding协议:


[cpp] view plaincopy

  1. extension Person {  

  2.   class HelperClass: NSObject, NSCoding {  

  3.       

  4.     var person: Person?  

  5.       

  6.     init(person: Person) {  

  7.       self.person = person  

  8.       super.init()  

  9.     }  

  10.       

  11.     class func path() -> String {  

  12.       let documentsPath = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomainMask.UserDomainMask, true).first  

  13.       let path = documentsPath?.stringByAppendingString("/Person")  

  14.       return path!  

  15.     }  

  16.       

  17.     required init?(coder aDecoder: NSCoder) {  

  18.       guard let firstName = aDecoder.decodeObjectForKey("firstName") as? String else { person = nil; super.init(); return nil }  

  19.       guard let laseName = aDecoder.decodeObjectForKey("lastName") as? String else { person = nil; super.init(); return nil }  

  20.         

  21.       person = Person(firstName: firstName, lastName: laseName)  

  22.         

  23.       super.init()  

  24.     }  

  25.       

  26.     func encodeWithCoder(aCoder: NSCoder) {  

  27.       aCoder.encodeObject(person!.firstName, forKey: "firstName")  

  28.       aCoder.encodeObject(person!.lastName, forKey: "lastName")  

  29.     }  

  30.   }  

  31. }  


发生了什么呢?我们在Person结构体当中增加了一个类,并使它遵守了NSCoding协议,这也意味着这个类需要实现init?(coder aDecoder: NSCoder)encodeWithCoder(aCoder: NSCoder)方法。这个类拥有一个类型为Person的属性,并且在encodeWithCoder(aCoder: NSCoder)方法中将这个结构体实例的值都进行了归档,同时在init?(coder aDecoder: NSCoder)中进行解档,并创建了一个新的person实例。

接下来要做的事就是向 Person 结构体的定义中增加归档和解档的方法:


[cpp] view plaincopy

  1. struct Person {  

  2.   let firstName: String  

  3.   let lastName: String  

  4.     

  5.   static func encode(person: Person) {  

  6.     let personClassObject = HelperClass(person: person)  

  7.       

  8.     NSKeyedArchiver.archiveRootObject(personClassObject, toFile: HelperClass.path())  

  9.   }  

  10.     

  11.   static func decode() -> Person? {  

  12.     let personClassObject = NSKeyedUnarchiver.unarchiveObjectWithFile(HelperClass.path()) as? HelperClass  

  13.   

  14.     return personClassObject?.person  

  15.   }  

  16. }  


在这段代码中,我们创建了一个HelperClass对象来帮助进行归档和解档。

这个结构体的使用方法应该是这样的:


[cpp] view plaincopy

  1. let me = Person(firstName: "Dominik", lastName: "Hauser")  

  2.       

  3. Person.encode(me)  

  4.       

  5. let myClone = Person.decode()  

  6.       

  7. firstNameLabel.text = myClone?.firstName  

  8. lastNameLabel.text = myClone?.lastName  


你可以在Github上找到完整的代码。


  • 项目经理 点击这里给我发消息
  • 项目经理 点击这里给我发消息
  • 项目经理 点击这里给我发消息
  • 项目经理 点击这里给我发消息