代码之家  ›  专栏  ›  技术社区  ›  Mumphus

向CoreData添加新版本

  •  1
  • Mumphus  · 技术社区  · 6 年前

    我有一个包含两个实体的coredata模型,一个实体与另一个实体有关系,(数据集---(对多个)数据点)。最初,我的关系是相反的,但最终我不需要它。我知道,如果我刚刚改变它,人们将不得不重新安装应用程序,删除他们的所有数据。所以我查找了如何更改它,但仍然允许旧模型兼容。

    我遵循了我发现的这些步骤,所以:

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

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

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

    未解决的错误错误域=您的“错误域代码=9999”未能 初始化应用程序保存的数据“ userinfo=nsLocalizedDescription=未能初始化 应用程序保存的数据,nsunderlyingerrror=0x1c06402d0错误 domain=nscocoaerrordomain code=134100“托管对象模型 用于打开持久存储的版本与该版本不兼容 用于创建持久存储。”

    我不确定我是错过了什么,还是做错了什么。我唯一改变的是,这种关系不再有一个在代码中甚至没有使用的逆关系。从我的阅读资料中我理解它的方式是,如果最新的型号不在设备上,它会寻找一个兼容的型号。

    1 回复  |  直到 6 年前
        1
  •  1
  •   digitalHound    6 年前

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

    像这样:

        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)")
                }
            }
        }
    }