基于这里的问题:Migrate iCloud data to Local store and stopping iCloud from still responding考虑将iCloud存储移动到本地存储并确保禁用iCloud通知,我有下一个我坚持的方案。
问题
现在,当用户在第一台设备上运行应用时,他们会在开始时询问是否要启用iCloud。如果选择是,则将数据迁移到iCloud。如果用户在第二个设备中连接,则会询问他们是否要在开始时使用iCloud,如果他们选择“是”,则会同步来自第一个设备的数据。
如果第二台设备上的用户决定不再使用iCloud,他们可以专门针对该设备导航并关闭应用内的iCloud。这在后端将其数据从iCloud存储迁移到本地存储。从用户的角度来看,他们的所有数据都存在(无论是存储在云中还是本地存储对用户来说都无关紧要)。
问题是当我迁移数据时,它成功迁移到本地iCloud存储并成功停止侦听iCloud通知。那个方面有效,这是我之前提出的堆栈溢出问题(Migrate iCloud data to Local store and stopping iCloud from still responding)。
具体问题是,当用户重新启动应用时,该应用现在为空,没有数据。这当然不是预期的效果。
为了完整起见,“迁移代码”基于另一个堆栈溢出问题,但为了方便我将它粘贴在这里:
- (void)migrateiCloudStoreToLocalStore {
NSLog(@"Migrate iCloudToLocalStore");
NSPersistentStore *store = self.persistentStoreCoordinator.persistentStores.lastObject;
//NSURL *storeURL = [[self applicationDocumentsDirectory] URLByAppendingPathComponent:@"Envylope.sqlite"];
NSURL *storeURL = [self.persistentStoreCoordinator.persistentStores.lastObject URL];
NSLog(@"Current Store URL (before iCloud to Local migration): %@", [storeURL description]);
NSDictionary *localStoreOptions = nil;
localStoreOptions = @{ NSPersistentStoreRemoveUbiquitousMetadataOption : @YES,
NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES};
NSPersistentStore *newStore = [self.persistentStoreCoordinator migratePersistentStore:store
toURL:storeURL
options:localStoreOptions
withType:NSSQLiteStoreType error:nil];
[self reloadStore:newStore];
}
- (void)reloadStore:(NSPersistentStore *)store {
NSLog(@"Reload Store");
NSURL *storeURL = [[self applicationDocumentsDirectory] URLByAppendingPathComponent:@"Envylope.sqlite"];
NSDictionary *localStoreOptions = nil;
localStoreOptions = @{ NSPersistentStoreRemoveUbiquitousMetadataOption : @YES,
NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES};
if (store) {
[self.persistentStoreCoordinator removePersistentStore:store error:nil];
}
[self.persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType
configuration:nil
URL:storeURL
options:localStoreOptions
error:nil];
storeURL = [self.persistentStoreCoordinator.persistentStores.lastObject URL];
NSLog(@"Current Store URL (after iCloud to Local migration): %@", [storeURL description]);
NSLog(@"Done reloading");
[[NSUserDefaults standardUserDefaults]setBool:YES forKey:@"MigratedFromiCloudToLocal"];
[[NSUserDefaults standardUserDefaults]synchronize];
所有的魔法都发生在persistentStoreCoordinator中..它很乱,但我需要让这个部分工作,然后才能清理它:
- (NSPersistentStoreCoordinator *)persistentStoreCoordinator
{
if (_persistentStoreCoordinator != nil) {
return _persistentStoreCoordinator;
}
NSURL *storeURL = [[self applicationDocumentsDirectory] URLByAppendingPathComponent:@"Envylope.sqlite"];
NSError *error = nil;
NSURL *iCloud = [[NSFileManager defaultManager] URLForUbiquityContainerIdentifier: nil];
NSDictionary *options = nil;
if ((iCloud) && (![[NSUserDefaults standardUserDefaults] boolForKey:@"iCloudOn"]) && (![[NSUserDefaults standardUserDefaults]boolForKey:@"OnLatestVersion"]))
{
NSLog(@"In the persistent store, iCloud is enabled in the app device but not yet in the app and not on the latest version");
options = @{ NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES};
}
else if ((iCloud) && ([[NSUserDefaults standardUserDefaults] boolForKey:@"iCloudOn"]) && ([[NSUserDefaults standardUserDefaults]boolForKey:@"OnLatestVersion"]))
{
NSLog(@"In the persistent store, iCloud is enabled, we're using iCloud from the Tutorial and we're on the latest version");
NSURL *cloudURL = [self grabCloudPath:@"iCloud"];
options = @{ NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES,
NSPersistentStoreUbiquitousContentURLKey: cloudURL, NSPersistentStoreUbiquitousContentNameKey: @"EnvyCloud"};
}
else if ((iCloud) && (![[NSUserDefaults standardUserDefaults] boolForKey:@"iCloudOn"]) && ([[NSUserDefaults standardUserDefaults]boolForKey:@"OnLatestVersion"]))
{
NSLog(@"In the persistent store, iCloud is enabled, we're on the latest version, btu we're not using icloud in the App");
options = @{ NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES};
}
else
{
NSLog(@"iCloud is just not enabled");
options = @{ NSMigratePersistentStoresAutomaticallyOption : @YES,
NSInferMappingModelAutomaticallyOption : @YES};
}
_persistentStoreCoordinator = [[NSPersistentStoreCoordinator alloc] initWithManagedObjectModel:[self managedObjectModel]];
if (![_persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeURL options:options error:&error]) {
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
//abort();
}
return _persistentStoreCoordinator;
}
因此,在用户在应用程序中启用iCloud,然后决定不再在此特定设备上安装iCloud的情况下,iCloud通知将被删除,商店将从iCloud Store迁移到本地商店,NSUserDefaults是组。在下一次启动应用程序时...第3个if语句在persistentStoreCoordinator
中评估为true,正确地说我们是在最新版本上,iCloud已启用(在设备中)但不在应用程序中,但是然后它显示应用程序绝对没有任何条目,有点像开始新鲜。
如何克服此问题并返回刚刚迁移的商店?
任何想法都会非常感激。
答案 0 :(得分:2)
应用启动时没有数据指向持久性商店协调员。
看起来对选项字典的评估很好,但在任何一种结果中,您总是使用相同的持久存储URL设置协调器。
同样,在迁移例程中,您将迁移持久性存储对象,但使用相同的URL。这不起作用,因为那是保存实际文件的地方。您需要本地存储和iCloud存储的不同URL,然后您可以正确地迁移数据并选择删除不再需要的URL(从协调器中删除存储后)。
使用相应的网址设置协调员,具体取决于您的评估结果 - 并且希望您应该在应用启动时看到您的数据。