向CoreData添加新版本

时间:2018-07-12 15:52:27

标签: ios xcode core-data

我有一个包含2个实体的coredata模型,一个实体与另一个实体有关系,(DataSet ---(To many)-> DataPoints)。本来我有这种关系,但是我最终不需要它。我知道,如果只是更改它,人们将不得不重新安装该应用程序,并删除所有数据。因此,我查看了如何更改它,但仍然允许旧模型兼容。

我遵循了在SO上找到的这些步骤:

  1. 选择* .xcdatamodelId
  2. 选择编辑器>添加模型版本
  3. 提供基于先前型号的版本名称
  4. 确保选择刚刚创建的新版本
  5. 为它提供一个新的标识符(在文件选择器中)
  6. 进行更改
  7. 选择* .xcdatamodelId并更改Model Version(在文件中 选择器)

我还选择了模型,并将“模型版本”更改为“ * v2”

但是当我在具有旧型号的设备上运行该应用程序时,一旦尝试创建数据,我就会收到错误消息:

  

未解决的错误错误域= YOUR_ERROR_DOMAIN代码= 9999“失败   初始化应用程序的已保存数据”   UserInfo = {NSLocalizedDescription =初始化失败   应用程序的已保存数据NSUnderlyingError = 0x1c06402d0 {Error   Domain = NSCocoaErrorDomain代码= 134100“托管对象模型   用于打开永久性存储的版本与该版本不兼容   用来创建持久性存储。”

我不确定是否错过了某件事或做错了什么。我唯一更改的是关系不再具有逆关系,该逆关系甚至不在代码中使用。我从阅读中了解到的方式是,如果设备上没有最新的型号,它将寻找兼容的型号。

1 个答案:

答案 0 :(得分:1)

设置持久性存储协调器时,请确保在设置时通过的选项字典中指定您希望它自动迁移数据(如果适用)。

类似的东西:

    do {
        try self?.psc?.addPersistentStore(
                       ofType: NSSQLiteStoreType, 
            configurationName: nil, 
                           at: url, 
                      options: [
                             NSMigratePersistentStoresAutomaticallyOption: true,
                             NSInferMappingModelAutomaticallyOption: true
                        ])
        print("Core Data Store setup")
    } catch {
        print("Error migrating store: \(error)")
    }

如果要为iOS 10+进行构建,另一个选择是利用NSPersistentContainer来为您管理上下文,模型和持久性存储协调器。

(我还没有尝试使用NSPersistentContainer进行迁移,但是我想我会通知您,以防它为您简化事情)。

这是一个使用NSPersistentContainer而不是带有持久性存储协调器的旧样式的示例核心数据堆栈,等等:

import CoreData

class CoreDataStack {

    // added in case you want to initialize the persistent container with a specific managed
    // object model via
    // let container = NSPersistentContainer.init(name: DataModel, managedObjectModel: managedObjectModel())
    internal func managedObjectModel() -> NSManagedObjectModel {
        let bundle = Bundle(for: AppDelegate.self)
        guard let url = bundle.url(forResource: "DataModel", withExtension: "momd") else {
            fatalError("Error loading model from bundle")
        }
        guard let mom = NSManagedObjectModel(contentsOf: url) else {
            fatalError("Error initializing mom from: \(url)")
        }

        return mom
    }

    internal lazy var persistentContainer: NSPersistentContainer = {
        let container = NSPersistentContainer(name: "DataModel")
        container.loadPersistentStores(completionHandler: { [weak self](storeDescription, error) in

            /*
             Typical reasons for an error here include:
             * The parent directory does not exist, cannot be created, or disallows writing.
             * The persistent store is not accessible, due to permissions or data protection when the device is locked.
             * The device is out of space.
             * The store could not be migrated to the current model version.
             Check the error message to determine what the actual problem was.
             */
            if let error = error {
                print("CoreData:  error \(error), \(String(describing: error._userInfo))")
            }
        })
        return container
    }()

    func performUITask(_ block: @escaping (NSManagedObjectContext) -> Void) {
        persistentContainer.viewContext.perform {
            block(self.persistentContainer.viewContext)
        }
    }

    func performBackgroundTask(_ block: @escaping (NSManagedObjectContext) -> Void) {
        persistentContainer.performBackgroundTask(block)
    }

    // MARK: - Core Data Saving support
    func saveContext () {
        let context = persistentContainer.viewContext
        if context.hasChanges {
            do {
                try context.save()
            } catch {
                // Replace this implementation with code to handle the error appropriately.
                // fatalError() causes the application to generate a crash log and terminate. You should not use this function in a shipping application, although it may be useful during development.
                let nserror = error as NSError
                fatalError("CoreData: Unresolved error \(nserror), \(nserror.userInfo)")
            }
        }
    }
}