订阅移动开发RSS CSDN首页> 移动开发

Swift中的结构体与NSCoding

发表于2015-08-27 15:30| 次阅读| 来源Swift and Painless| 0 条评论| 作者Dominik Hauser

摘要:Swift中的结构体不遵守NSCoding协议,NSCoding只适用于继承自NSObject的类,可是结构体在Swift中的地位与使用频率都非常高,于是,本文作者灵光一闪想到了一个曲线救国的解决方法。

CSDN移动将持续为您优选移动开发的精华内容,共同探讨移动开发的技术热点话题,涵盖移动应用、开发工具、移动游戏及引擎、智能硬件、物联网等方方面面。如果您想投稿、参与内容翻译工作,或寻求近匠报道,请发送邮件至tangxy#csdn.net(请把#改成@)。 


本文出自:Swift and Painless,作者:Dominik Hauser,译文出自:SwiftGG,译者:宜东

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

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

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

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

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

struct Person {
  let firstName: String
  let lastName: String
}

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

extension Person {
  class HelperClass: NSObject, NSCoding {
    
    var person: Person?
    
    init(person: Person) {
      self.person = person
      super.init()
    }
    
    class func path() -> String {
      let documentsPath = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomainMask.UserDomainMask, true).first
      let path = documentsPath?.stringByAppendingString("/Person")
      return path!
    }
    
    required init?(coder aDecoder: NSCoder) {
      guard let firstName = aDecoder.decodeObjectForKey("firstName") as? String else { person = nil; super.init(); return nil }
      guard let laseName = aDecoder.decodeObjectForKey("lastName") as? String else { person = nil; super.init(); return nil }
      
      person = Person(firstName: firstName, lastName: laseName)
      
      super.init()
    }
    
    func encodeWithCoder(aCoder: NSCoder) {
      aCoder.encodeObject(person!.firstName, forKey: "firstName")
      aCoder.encodeObject(person!.lastName, forKey: "lastName")
    }
  }
}

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

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

struct Person {
  let firstName: String
  let lastName: String
  
  static func encode(person: Person) {
    let personClassObject = HelperClass(person: person)
    
    NSKeyedArchiver.archiveRootObject(personClassObject, toFile: HelperClass.path())
  }
  
  static func decode() -> Person? {
    let personClassObject = NSKeyedUnarchiver.unarchiveObjectWithFile(HelperClass.path()) as? HelperClass

    return personClassObject?.person
  }
}

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

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

let me = Person(firstName: "Dominik", lastName: "Hauser")
    
Person.encode(me)
    
let myClone = Person.decode()
    
firstNameLabel.text = myClone?.firstName
lastNameLabel.text = myClone?.lastName

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


预告: 2015中国移动开发者大会(MDCC 2015)将于10月15-16日在北京新云南皇冠假日酒店召开。大会特设五大技术专场:平台与技术iOS、平台与技术Android、产品与设计、游戏开发、企业移动化。此外,大会更是首次举办国内极具权威影响力的IoT技术峰会,特设硬件开发技术与虚拟现实两大专场。大会将聚集国内最具实力的产品技术团队,与开发者一道进行最前沿的探讨与交流。

第一时间掌握最新移动开发相关信息和技术,请关注mobilehub公众微信号(ID: mobilehub)。


  • CSDN官方微信
  • 扫描二维码,向CSDN吐槽
  • 微信号:CSDNnews
程序员移动端订阅下载

微博关注

相关热门文章