我有一个OSX和iOS的应用程序,都使用iCloud在它们之间共享核心数据。当我在iPhone上进行更改时,我可以在OSX应用程序中看到它们,并在两个应用程序中调用NSPersistentStoreDidImportUbiquitousContentChangesNotification
,因此iOS - > Mac效果很好。但是当我在Mac应用程序中进行一些更改时,我无法在iPhone中看到它们。 iPhone永远不会调用NSPersistentStoreDidImportUbiquitousContentChangesNotification
。我能改变Mac的唯一方法是在iPhone上做一些改变。然后我可以看到所有的变化。
两个项目中的iCloud集成是一样的。核心数据数据模型是相同的。权利也是一样的。
为什么在我在iPhone上进行一些更改之前iPhone没有改变Mac?有什么想法吗?
另一个问题
在我的应用程序中,当用户启用iCloud时,我会检查iCloud上是否已有文件。如果文件存在,则用户可以选择从iCloud文件还原核心数据或从本地存储启动新副本。 要开始新的副本,我使用以下代码:
- (void)startNewCopy
{
[self removeICloudStore];
[self moveStoreToIcloud];
}
- (void)removeICloudStore
{
BOOL result;
NSError *error;
// Now delete the iCloud content and file
result = [NSPersistentStoreCoordinator removeUbiquitousContentAndPersistentStoreAtURL:[self icloudStoreURL]
options:[self icloudStoreOptions]
error:&error];
if (!result)
{
NSLog(@"Error removing store");
}
else
{
NSLog(@"Core Data store removed.");
// Now delete the local file
[self deleteLocalCopyOfiCloudStore];
}
}
- (void)deleteLocalCopyOfiCloudStore {
// We need to get the URL to the store
NSError *error = nil;
[[NSFileManager defaultManager] removeItemAtURL:[self localUbiquitySupportURL] error:&error];
}
- (void)moveStoreToIcloud
{
// Open the store
NSPersistentStore *sourceStore = [[_persistentStoreCoordinator persistentStores] firstObject];
if (!sourceStore)
{
NSLog(@" failed to add old store");
}
else
{
NSLog(@" Successfully added store to migrate");
NSError *error;
id migrationSuccess = [_persistentStoreCoordinator migratePersistentStore:sourceStore toURL:[self icloudStoreURL] options:[self icloudStoreOptions] withType:NSSQLiteStoreType error:&error];
if (migrationSuccess)
{
NSLog(@"Store migrated to iCloud");
_persistentStoreCoordinator = nil;
_managedObjectContext = nil;
// Now delete the local file
[self deleteLocalStore];
}
else
{
NSLog(@"Failed to migrate store: %@, %@", error, error.userInfo);
}
}
}
- (void)deleteLocalStore
{
NSError *error = nil;
[[NSFileManager defaultManager] removeItemAtURL:[self localStoreURL] error:&error];
}
从单个设备执行此操作似乎都运行良好,但是当我尝试使用多个设备时,我遇到了一些错误。
示例:
我有两个设备。第一个没有连接到iCloud,第二个连接到iCloud。
在启用iCloud的第一个设备中,我选择startNewCopy
,然后在第二个设备中出现此错误:
CoreData:Ubiquity:Librarian在开始下载时返回了一个严重错误错误Domain = BRCloudDocsErrorDomain Code = 5“操作无法完成。(BRCloudDocsErrorDomain错误5 - URL上没有文档)”
错误之前永远不会调用NSPersistentStoreCoordinatorStoresDidChangeNotification
。
这是我的通知代码:
- (void)processStoresDidChange:(NSNotification *)notification
{
NSLog(@"processStoresDidChange");
// Post notification to trigger UI updates
// Check type of transition
NSNumber *type = [notification.userInfo objectForKey:NSPersistentStoreUbiquitousTransitionTypeKey];
//NSLog(@" userInfo is %@", notification.userInfo);
//NSLog(@" transition type is %@", type);
if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeInitialImportCompleted) {
NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeInitialImportCompleted");
} else if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeAccountAdded) {
NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeAccountAdded");
} else if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeAccountRemoved) {
NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeAccountRemoved");
} else if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeContentRemoved) {
NSLog(@" transition type is NSPersistentStoreUbiquitousTransitionTypeContentRemoved");
}
[[NSOperationQueue mainQueue] addOperationWithBlock:^ {
if (type.intValue == NSPersistentStoreUbiquitousTransitionTypeContentRemoved) {
[self deleteLocalStore];
_persistentStoreCoordinator = nil;
_managedObjectContext = nil;
NSLog(@" iCloud store was removed! Wait for empty store");
}
// Refresh user Interface
[[NSNotificationCenter defaultCenter] postNotificationName:@"notiIcloud" object:@"storeChanged"];
}];
}
如何检测iCloud内容已被删除并避免上述错误?
答案 0 :(得分:0)
在我看来,你可能没有使用适当的接口。
您可以在不使用NSPersistentStore
方法的情况下将文件移入和移出iCloud。
通常,相应的操作也必须包含在NSFileCoordinator
电话中。
这是我在iOS上做的事情。 toLocal
表示移入或移出本地(设备)存储:
//
/// Move file between stores (assumed no name clash).
/// Return new URL.
/// Note: cloud file moves are asynchronous.
///
- (NSURL *)moveStoreFile:(NSURL *)url toRootURL:(NSURL *)rootURL toLocal:(BOOL)toLocal
{
NSURL *newURL = rootURL;
if (!toLocal)
newURL = [newURL URLByAppendingPathComponent:@"Documents"];
newURL = [newURL URLByAppendingPathComponent:url.lastPathComponent];
[self backupFile:url isLocal:!toLocal completionHandler:
^(BOOL success) {
MRLOG(@"MOVE %s %@ to %s %@", toLocal? "icloud" : "local", [url lastPathComponent],
toLocal? "local" : "icloud", [newURL lastPathComponent]);
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
^{
NSError *error;
// setUbiquitous does file coordination
BOOL msuccess = [[NSFileManager defaultManager]
setUbiquitous:!toLocal itemAtURL:url destinationURL:newURL error:&error];
dispatch_async(dispatch_get_main_queue(),
^{
if (!msuccess)
MRLOG(@"move failed: %@", error);
[self.delegate moveStoreFileCompletedWithStatus:success error:error];
});
});
}];
return newURL;
}
删除需要明确的文件协调:
///
/// Purge backup file (really delete it).
/// Note: cloud deletes are asynchronous.
///
- (void)purgeFile:(NSURL *)url isLocal:(BOOL)isLocal
{
MRLOG(@"PURGE %s %@", isLocal? "local" : "icloud", [url lastPathComponent]);
if (isLocal)
[self removeFile:url];
else
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0),
^{
NSError *coordinationError;
NSFileCoordinator *coordinator = [[NSFileCoordinator alloc] initWithFilePresenter:nil];
[coordinator coordinateWritingItemAtURL:url options:NSFileCoordinatorWritingForDeleting
error:&coordinationError
byAccessor:
^(NSURL* writingURL)
{
[self removeFile:writingURL];
}];
if (coordinationError) {
MRLOG(@"coordination error: %@", coordinationError);
[(SSApplication *)[SSApplication sharedApplication] fileErrorAlert:coordinationError];
}
});
}
要检测您需要使用NSMetadataQuery
的文件的更改,并为NSMetadataQueryDidUpdateNotification
添加观察者。
答案 1 :(得分:0)
我有一个类似的问题,将iOS与我的Mac同步。
此错误5实际上意味着您的商店存在问题。
您可以尝试以下步骤:
您可以通过致电重置您的iCloud内容(抱歉,我在Swift中拥有它):
try! NSPersistentStoreCoordinator.removeUbiquitousContentAndPersistentStoreAtURL(storeURL, options: self.options)
当Model(xcdatamodeld文件)发生更改且现有数据尚未清除时,会发生此问题。您最终得到一个与旧模型下生成的数据相关联的新模型,并且您尝试跨设备同步该混合...
将来,要避免与模型相关的问题,请查看核心数据模型版本控制和数据迁移。 https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/CoreDataVersioning/Articles/Introduction.html
希望它能解决你的问题。