首页 > 解决方案 > 保存不断附加的自定义对象数组

问题描述

我对 Swift 和一般编码比较陌生。我目前正在努力磨练自己的技能,但正在整理一个简单的提醒应用程序。在我把故事板放在一起之前,我试图让后端工作,但我有必要的故事板元素来测试我的系统是否可以工作。

基本上我正在尝试保存一个包含自定义对象的数组,但是这个数组被附加到用户完成的每个提醒添加中。这样每次应用打开时,数组都会包含上次的提醒。

这是我到目前为止创建和附加列表的代码;

func createReminder() {
    let reminderAdd = Reminder(chosenReminderDescription: textRetrieve.text!, chosenReminderLength: 1)
    reminderList.append(reminderAdd)

    dump(reminderList)
}

这是目标代码;

class Reminder {

    var reminderDescription = "Require initalisation."
    var reminderLength = 1 // in days

    init (chosenReminderDescription: String, chosenReminderLength: Int) {
        reminderDescription = chosenReminderDescription
        reminderLength = chosenReminderLength
    }

}

我将如何保存数组?

编辑: 这是我到目前为止添加的内容。

    override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib.
    let reminderAdd = Reminder(chosenReminderDescription: "Placeholder test", chosenReminderLength: 1)
    reminderList.append(reminderAdd)

    let appDelegate = UIApplication.shared.delegate as! AppDelegate
    let context = appDelegate.persistentContainer.viewContext

    let entity = NSEntityDescription.entity(forEntityName: "Tasks", in: context)
    let newTask = NSManagedObject(entity: entity!, insertInto: context)

    newTask.setValue(reminderList, forKey: "taskName")

    do {
        try context.save()
    } catch {
        print("Failed saving")
    }

    let request = NSFetchRequest<NSFetchRequestResult>(entityName: "Tasks")
    //request.predicate = NSPredicate(format: "age = %@", "12")
    request.returnsObjectsAsFaults = false
    do {
        let result = try context.fetch(request)
        for data in result as! [NSManagedObject] {
            print(data.value(forKey: "taskName"))
        }

    } catch {

        print("Failed")
    }

我遇到了崩溃,我似乎还无法调试它。我相信这条线导致崩溃,因为当我删除它时,应用程序启动正常。

let reminderAdd = Reminder(chosenReminderDescription: "Placeholder test", chosenReminderLength: 1)
    reminderList.append(reminderAdd)

有任何想法吗?

编辑 2: 数据模型

那是数据模型,我不完全确定您将对象变成可编码的意思。再次感谢。

编辑 3:

ViewDidLoad

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib

    let appDelegate = UIApplication.shared.delegate as! AppDelegate
    let context = appDelegate.persistentContainer.viewContext

    let entity = NSEntityDescription.entity(forEntityName: "Tasks", in: context)
    let newTask = Tasks(entity: entity!, insertInto: context)

    newTask.setValue(reminderList, forKey: "taskName")

    do {
        try context.save()
    } catch {
        print("Failed saving")
    }

    let request = NSFetchRequest<NSFetchRequestResult>(entityName: "Tasks")

    //request.predicate = NSPredicate(format: "age = %@", "12")
    request.returnsObjectsAsFaults = false
    do {
        let result = try context.fetch(request)
        for data in result as! [Tasks] {
            print(data.value(forKey: "taskName"))

        }

    } catch {

        print("Failed")
    }

    dump(reminderList)
}

标签: arraysswiftsave

解决方案


您可以使用 CoreData 创建一个实例并将其存储为内部数据库。这些是一些很好的教程:

https://medium.com/xcblog/core-data-with-swift-4-for-beginners-1fc067cca707

https://www.raywenderlich.com/7569-getting-started-with-core-data-tutorial

编辑 2

正如您在这张图片中看到的那样, https: //ibb.co/f1axcA 我在 coreData 中的列表是 [Notifica] 类型的,对象 Notifica 的数组也是如此,要实现可编码,您应该执行以下操作

public class Notifica: NSObject, NSCoding {

public required init?(coder aDecoder: NSCoder) {
    self.id = aDecoder.decodeObject(forKey: "id") as? Double
    self.type = aDecoder.decodeObject(forKey: "type") as? String
    self.idEvent = aDecoder.decodeObject(forKey: "idEvent") as? Int
    self.contactPerson = aDecoder.decodeObject(forKey: "contactPerson") as? People
    self.title = aDecoder.decodeObject(forKey: "title") as? String
    self.date = aDecoder.decodeObject(forKey: "date") as? String
}

public func encode(with aCoder: NSCoder) {
    aCoder.encode(id, forKey: "id")
    aCoder.encode(type, forKey: "type")
    aCoder.encode(idEvent, forKey: "idEvent")
    aCoder.encode(contactPerson, forKey: "contactPerson")
    aCoder.encode(title, forKey: "title")
    aCoder.encode(date, forKey: "date")
}

ecc..

另一件事是不要调用 NSManagedObject 并传递实体,但您应该像在 dataModel 中调用的那样命名 Tasks,如果您在 xcode 上键入 Tasks,它将为您找到创建的 NSManagedObject,然后您可以设置 taskName 的值

编辑 3

" <Simple_Reminders.Reminder: 0x60400046da40>" 表示存在 Reminder 对象!所以你救了它!提醒有两个变量:-reminderDescription 和 -reminderLength,所以改变你的代码

do {
    let result = try context.fetch(request)
    for data in result as! [Tasks] {
        print(data.value(forKey: "taskName"))

    }

} catch {

    print("Failed")
}

有了这个

do {
    let result = try context.fetch(request)
    for data in result as! [Tasks] {
        print(data.value(forKey: "taskName"))
        if let reminders = data.value(forKey: "taskName") as? [Reminder] {
            for reminder in reminders {
                // Now you have your single object Reminder and you can print his variables
                print("Your reminder description is \(reminder. reminderDescription), and his length is \(reminder. reminderLength))"
            }
        }
    }

} catch {

    print("Failed")
}

推荐阅读