ios - 向 CoreData 添加新版本
问题描述
我有一个包含 2 个实体的 coredata 模型,一个实体与另一个实体有关系,(DataSet---(To many)-->DataPoints)。最初我有这种关系有一个逆,但我最终不需要它。我知道如果我只是更改它,人们将不得不重新安装该应用程序,删除他们的所有数据。所以我查找了如何更改它但仍然允许旧模型兼容。
我按照我在 SO 上找到的这些步骤进行操作:
- 选择 *.xcdatamodelId
- 选择编辑器 > 添加模型版本
- 根据以前的模型提供版本名称
- 确保选择刚刚创建的新版本
- 给它一个新的标识符(在文件选择器中)
- 进行更改
- 选择 *.xcdatamodelId 并更改模型版本(在文件选择器中)
我还选择了模型并将“模型版本”更改为“*v2”
但是,当我在使用旧型号的设备上运行应用程序时,一旦它尝试创建数据,我就会收到错误消息:
未解决的错误 Error Domain=YOUR_ERROR_DOMAIN Code=9999 "无法初始化应用程序保存的数据" UserInfo={NSLocalizedDescription=无法初始化应用程序保存的数据, NSUnderlyingError=0x1c06402d0 {Error Domain=NSCocoaErrorDomain Code=134100 "托管对象模型版本用于打开持久存储与用于创建持久存储的不兼容。”
我不确定我是错过了什么还是做错了什么。我唯一改变的是关系不再有逆,甚至没有在代码中使用。我从阅读中理解的方式是,如果设备上没有最新型号,它会寻找兼容的型号。
解决方案
当您设置持久存储协调器时,请确保指定您希望它自动迁移您的数据(如果适用)在设置时传递的选项字典中。
像这样的东西:
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)")
}
}
}
}
推荐阅读
- reactjs - 将 React 项目克隆到新 PC 上
- python-3.x - Softmax 没有给出每个类别的概率
- kubernetes - 如何在不传递脚本的情况下使用 Kubernetes 多个命令和参数
- r - tidyverse 中的非 equi 连接
- json - WebAPI:JSON ReferenceHandler.Preserve
- javascript - Discord.js - TypeError:无法读取未定义的属性“声音”。有人可以告诉我我做错了什么吗?
- swiftui - 如何缩进快速左对齐的文本和图像?
- spartacus-storefront - 更改语言 languageService.getAll() 数据没有改变
- applescript - 在 Big Sur 中切换灰度的 Automator 脚本
- google-chrome-extension - 特定 url 的内容脚本仅在刷新时运行,而不在导航时运行