在我的应用程序中,我尝试使用数据持久性保存数据,但是我收到此错误: 必须拨打超类“指定日期”的指定初始值设定项。错误。
这是我的代码:
class Day: NSObject, NSCoding { // NEMA NSCODING I NSOBJECT
var dayName: String
var subjects: [Subject]?
init(dayName: String) {
self.dayName = dayName
}
struct PropertyKey {
static var dayName = "dayName"
static var subjects = "subjects"
}
func encode(with aCoder: NSCoder) {
aCoder.encode(dayName, forKey: PropertyKey.dayName)
aCoder.encode(subjects, forKey: PropertyKey.subjects)
}
// Archiving paths
static let DocumentsDirectory = FileManager().urls(for: .documentDirectory, in: .userDomainMask).first!
static let ArchiveURL = DocumentsDirectory.appendingPathComponent("workingDays")//-namesto meals
required convenience init?(coder aDecoder: NSCoder) {
guard let dayName = aDecoder.decodeObject(forKey: PropertyKey.dayName) as? String else {
os_log("Unable to decode the dayName for a Day object.", log: OSLog.default, type: .debug)
return nil
}
let subjects = aDecoder.decodeObject(forKey: PropertyKey.subjects) as? [Subject]
self.init(dayName: dayName)
}
}
这是另一个班级:
class Subject: Day {
var subjectName: String
var startsAt: String?
init(dayName: String,subjectName: String) {
self.subjectName = subjectName
super.init(dayName: dayName)
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder) // here I get the error
}
}
我是否会以这种方式保存数据,仅在Day类中实现数据持久性,为什么我会收到该错误? 我是初学者,而且我是基于这个苹果文档做的 -
非常感谢任何帮助。
答案 0 :(得分:2)
根据Required Initializers的定义:
在类初始值设定项的定义之前编写
required
修饰符,以指示该类的每个子类都必须 实现初始化程序:覆盖所需的指定初始化程序时,您不会编写
override
修饰符:
在您的情况下,您创建了超级类convenience initialiser
的{{1}},因此子类必须立即实现它:
代码更正 - :
required
答案 1 :(得分:2)
问题是由self.init
init?(coder
中Day
的{{1}}调用引起的。该调用使初始化器convenient
和子类无法满足调用指定初始化器的要求。
解决方案是直接初始化属性
required init?(coder aDecoder: NSCoder) {
self.dayName = aDecoder.decodeObject(forKey: PropertyKey.dayName) as! String
self.subjects = aDecoder.decodeObject(forKey: PropertyKey.subjects) as? [Subject]
}
顺便说一句:您将总是 dayName
编码为非可选字符串,因此在解码时永远不会是nil
。 guard
没用。
在子类中,您可能需要添加代码来解码/解码子类的属性,并调用super
以考虑超类的属性。
class Subject: Day {
var subjectName: String
var startsAt: String?
init(dayName: String, subjectName: String) {
self.subjectName = subjectName
super.init(dayName: dayName)
}
override func encode(with aCoder: NSCoder) {
super.encode(with: aCoder)
aCoder.encode(subjectName, forKey: "subjectName")
aCoder.encode(startsAt, forKey: "startsAt")
}
required init?(coder aDecoder: NSCoder) {
subjectName = aDecoder.decodeObject(forKey: "subjectName") as! String
startsAt = aDecoder.decodeObject(forKey: "startsAt") as? String
super.init(coder: aDecoder)
}
}
在其超类中使用子类作为属性的问题是另一个故事