Swift中的结构体与NSCoding
程序员文章站
2022-04-30 22:14:20
...
本文出自:Swift and Painless,作者:Dominik Hauser,译文出自:SwiftGG,译者:宜东
正如大家所知,Swift中的结构体不遵守NSCoding协议。NSCoding只适用于继承自NSObject的类。可是结构体在Swift中的地位与使用频率都非常高,因此,我们需要一个能将结构体的实例归档和解档的方法。
Janie写过在Sonoplot工作时,他们团队对此的解决方法。
简而言之,他们定义了一个拥有两个方法的协议:一个方法可以从结构体当中获得一个NSDictionary,另一个方法可以使用NSDictionary来初始化一个结构体。接着,再使用NSKeyedArchiver对这个NSDictionary进行序列化。这个方案的优雅之处在于,只要遵守了这个协议的结构体都可以进行序列化。
我最近灵光一闪,想到了另一种解决方案。尽管我已经实现了这种方案,并且使用它开发过几个小项目,但是我还是不确定这是不是一个好的方案。这个方法的优雅程度无法与上面提到的方法相提并论。然而我还是将它写出来,让读者自己来进行判断。
假设我们有一个person结构体:
我们不能使这个结构体遵守NSCoding协议,但是我们可以在结构体当中增加一个类的定义,使这个类来遵守NSCoding协议:
发生了什么呢?我们在Person结构体当中增加了一个类,并使它遵守了NSCoding协议,这也意味着这个类需要实现init?(coder aDecoder: NSCoder)和encodeWithCoder(aCoder: NSCoder)方法。这个类拥有一个类型为Person的属性,并且在encodeWithCoder(aCoder: NSCoder)方法中将这个结构体实例的值都进行了归档,同时在init?(coder aDecoder: NSCoder)中进行解档,并创建了一个新的person实例。
接下来要做的事就是向 Person 结构体的定义中增加归档和解档的方法:
在这段代码中,我们创建了一个HelperClass对象来帮助进行归档和解档。
这个结构体的使用方法应该是这样的:
你可以在Github上找到完整的代码。
正如大家所知,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上找到完整的代码。
上一篇: Linux下启动MongoDB
下一篇: fastjson序列化相关
推荐阅读
-
初步剖析C语言编程中的结构体
-
数据结构与算法(3)- C++ STL与java se中的vector
-
python中定义结构体的方法
-
iOS的UI开发中Modal的使用与主流应用UI结构介绍
-
静态成员函数和(CPP与C结构体的区别)
-
GO-结构体(类似python中的类)
-
Python cookbook(数据结构与算法)从序列中移除重复项且保持元素间顺序不变的方法
-
iOS开发之OC与swift开发混编教程,代理的相互调用,block的实现。OC调用Swift中的代理, OC调用Swift中的Block 闭包
-
C++中自定义结构体或类作为关联容器的键
-
C语言中结构体与指针的若干问题(在数据结构中的应用)