1

我正在尝试让 NSPersistentStoreCoordinator 管理多个持久存储的删除和插入。到目前为止,我已经设法为 PSC 配置了两个商店,并且我已经能够通过指定其索引来删除其中一个商店。

像这样……</p>

NSPersistentStore *store = [[self.persistentStoreCoordinator persistentStores] objectAtIndex:0];

        if (![self.persistentStoreCoordinator removePersistentStore:store error:&error]) {

            NSLog(@"Unresolved error %@, %@", error, [error userInfo]);

            abort();

        }

        [[NSFileManager defaultManager] removeItemAtURL:store.URL error:&error];

但是我发现当我将商店重新添加到 PSC 时,索引值不正确,并且无法使用现有的类方法指定。这样做的后果是新数据被下载并添加到错误的存储中。

有人对如何做到这一点有任何建议吗?

背景(更新)

使用两个持久存储的原因是我可以为将通过网络下载的两个 xml 文档指定一个唯一的存储。由于这两个文件都比较大,我希望减少网络流量。因此,检查是否有任何文件已被修改。如果有,则删除相应的持久存储并添加新存储。这时候问题就开始了。添加新存储总是将其添加到 Persistent Stores Array 的末尾。当将数据与 MOC 合并时,这似乎会在存储中造成不匹配。

代码 这是我迄今为止尝试过的,它删除并添加了持久存储,但新数据被添加到错误的存储中。

static NSString * const kLastDateStoreUpdateKey = @"eventLastStoreUpdateKey";

- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {

NSString *last_modified = [NSString stringWithFormat:@"%@",[[(NSHTTPURLResponse *)response allHeaderFields] objectForKey:@"Last-Modified"]];

NSDateFormatter *dFormatter = [[NSDateFormatter alloc] init];
[dFormatter setDateFormat:@"EEE, dd MMM yyyy HH:mm:ss zzz"];
[dFormatter setLocale:[[NSLocale alloc] initWithLocaleIdentifier:@"en_GB"]];
[dFormatter setTimeZone:[NSTimeZone timeZoneWithAbbreviation:@"GMT"]];

dateModified = [dFormatter dateFromString:last_modified];
NSDate *previousDate = [[NSUserDefaults standardUserDefaults] objectForKey:kLastDateStoreUpdateKey];

if (!previousDate || [previousDate compare:dateModified] != NSOrderedSame) {

    [self.managedObjectContext lock];
    [self.managedObjectContext reset];


    if ([[NSFileManager defaultManager] fileExistsAtPath:self.persistentStorePath]) {
        NSError *error = nil;

        NSArray *stores = [self.persistentStoreCoordinator persistentStores];
        NSURL *storeUrls = [NSURL fileURLWithPath:persistentStorePath];
        for (NSPersistentStore *store in stores){

            if ([[store.URL absoluteURL] isEqual:[storeUrls absoluteURL]]) {
                if (![self.persistentStoreCoordinator removePersistentStore:store error:&error]) {
                    NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
                    abort();
                }
                [[NSFileManager defaultManager] removeItemAtURL:store.URL error:&error];
                NSLog(@"Check store removed %@", [self.persistentStoreCoordinator persistentStores]);
            }
        }
    }

   NSURL *storeUrl = [NSURL fileURLWithPath:persistentStorePath];
    NSError *error = nil;
    [self.persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:nil URL:storeUrl options:nil error:&error];
    if (error) {
        NSLog(@"event error %@ %@",error, [[self.persistentStoreCoordinator persistentStores] objectAtIndex:0]);
    }
    NSLog(@"Check store added %@",self.persistentStoreCoordinator.persistentStores);

    [self.managedObjectContext unlock];

}else {
    [self cancelDownload];
    NSLog(@"event cancel %@ %@ %@",previousDate, dateModified, [self.persistentStoreCoordinator persistentStores]);
}
}

已解决 正如 Hunter 在下面指出的那样,我的 PSC 配置设置为 nil。所以数据不会被添加到每个单独的持久存储中。当我在核心数据模型中创建配置并针对相关实体时,我将 PSC 设置为具有该配置。然后它按预期工作。见下文。

   NSURL *storeUrl = [NSURL fileURLWithPath:persistentStorePath];
NSError *error = nil;
[self.persistentStoreCoordinator addPersistentStoreWithType:NSSQLiteStoreType configuration:@"ConfigEvent" URL:storeUrl options:nil error:&error];
if (error) {
    NSLog(@"event error %@ %@",error, [[self.persistentStoreCoordinator persistentStores] objectAtIndex:0]);
}
NSLog(@"Check store added %@",self.persistentStoreCoordinator.persistentStores);
4

2 回答 2

2

如果您确实需要这两个商店 [我无法从您的解释中判断这是否真的是最好的方法],您可能希望通过属性或 ivar 引用它们,而不是依赖数组。

或者,您可以遍历 PSC 的 stores 数组并通过检查 URL 来识别每一个。

更新:

如果您在将特定数据存储在不同的 NSPersistentStore 中时遇到问题,请查看 Core Data 配置。这允许您告诉 Core Data 将特定实体放在特定的持久存储中。

您在模型中以及添加持久存储时指定配置。更多信息:https ://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/CoreData/Articles/cdMOM.html

于 2012-05-09T15:13:46.523 回答
0

您应该能够通过一个持久存储协调器和两个托管对象上下文来实现这一点。如果您遇到影响性能的锁定,您只需要考虑两个持久存储协调器的复杂性。

记住:最简单的事情 (tm)

于 2012-05-09T13:26:21.777 回答