必须拨打超类“指定日期”的指定初始值设定项。错误

时间:2017-12-02 16:44:59

标签: swift data-persistence

在我的应用程序中,我尝试使用数据持久性保存数据,但是我收到此错误: 必须拨打超类“指定日期”的指定初始值设定项。错误。

这是我的代码:

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类中实现数据持久性,为什么我会收到该错误? 我是初学者,而且我是基于这个苹果文档做的 -

https://www.google.com/url?sa=t&rct=j&q=&esrc=s&source=web&cd=12&cad=rja&uact=8&ved=0ahUKEwjWoIHk4uvXAhUHaFAKHbQFCHcQFghbMAs&url=https%3A%2F%2Fdeveloper.apple.com%2Flibrary%2Fcontent%2Freferencelibrary%2FGettingStarted%2FDevelopiOSAppsSwift%2FPersistData.html&usg=AOvVaw24H5Uo4RyRY7NYIlztlgCj

非常感谢任何帮助。

2 个答案:

答案 0 :(得分:2)

根据Required Initializers的定义:

  

在类初始值设定项的定义之前编写required修饰符,以指示该类的每个子类都必须   实现初始化程序

     

覆盖所需的指定初始化程序时,您不会编写override修饰符:

在您的情况下,您创建了超级类convenience initialiser的{​​{1}},因此子类必须立即实现它:

代码更正 - :

required

答案 1 :(得分:2)

问题是由self.init init?(coderDay的{​​{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编码为非可选字符串,因此在解码时永远不会是nilguard没用。

在子类中,您可能需要添加代码来解码/解码子类的属性,并调用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)
    }
}

在其超类中使用子类作为属性的问题是另一个故事