34

我有一个非常烦人的问题,我似乎无法解决。

当我发送一条保存到核心数据的消息时,我有一个视图,完成后它会向数据库询问随机消息(句子)并将其保存到数据库中的另一行。

如果我对最后一部分进行硬编码,而不从数据库中获取数据,它工作得很好而且很花哨,但是一旦我从数据库中获取随机行,它就会发疯。

在我的 AppDelegate.m 中:

- (void)save {
    NSAssert(self.context != nil, @"Not initialized");
    NSError *error = nil;
    BOOL failed = [self.context hasChanges] && ![self.context save:&error];
    NSAssert1(!failed,@"Save failed %@",[error userInfo]);
}

- (NSString*)selectRandomSentence
{
    NSFetchRequest *request = [[NSFetchRequest alloc] init];
    NSEntityDescription *entity = [NSEntityDescription entityForName:@"Sentences" inManagedObjectContext:self.managedObjectContext];
    [request setEntity:entity];

    NSError *error = nil;
    NSUInteger count = [self.context countForFetchRequest:request error:&error];

    NSUInteger offset = count - (arc4random() % count);
    [request setFetchOffset:offset];
    [request setFetchLimit:1];

    NSArray *sentenceArray = [self.context executeFetchRequest:request error:&error];

    [request release];

    return [[sentenceArray objectAtIndex:0] sentence];
}

- (NSManagedObjectContext *)context {

    if (_managedObjectContext != nil)
        return _managedObjectContext;

    NSPersistentStoreCoordinator *coordinator = [self coordinator];
    if (coordinator != nil) {
        _managedObjectContext = [[NSManagedObjectContext alloc] init];
        [_managedObjectContext setPersistentStoreCoordinator:coordinator];
    }

    return _managedObjectContext;
}

在我的 ChatController.m 中:

- (void)didRecieveMessage:(NSString *)message
{
    [self addMessage:message fromMe:NO];
}

#pragma mark -
#pragma mark SendControllerDelegate

- (void)didSendMessage:(NSString*)text {
    [self addMessage:text fromMe:YES];
}

#pragma mark -
#pragma mark Private methods

- (void)responseReceived:(NSString*)response {
    [self addMessage:response fromMe:NO];
}

- (void)addMessage:(NSString*)text fromMe:(BOOL)fromMe {
    NSAssert(self.repository != nil, @"Not initialized");
    Message *msg = [self.repository messageForBuddy:self.buddy];
    msg.text = text;
    msg.fromMe = fromMe;

    if (fromMe)
    {
        [self.bot talkWithBot:text];
    }

    [self.repository asyncSave];

    [self.tableView reloadData];
    [self.tableView scrollToRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:[self.buddy.messages count] - 1] atScrollPosition:UITableViewScrollPositionBottom animated:YES];
}

在我的 OfflineBot.m 中:

- (void)talkWithBot:(NSString *)textFromMe
{
    AppDelegate *delegate = [[UIApplication sharedApplication] delegate];
    [self didRecieveMessage:[delegate selectRandomSentence]];
}

- (void)didRecieveMessage:(NSString *)message
{
    if ([self.delegate respondsToSelector:@selector(didRecieveMessage:)])
        [self.delegate didRecieveMessage:message];
}

存储库.m

- (Message*)messageForBuddy:(Buddy*)buddy {
    Message *msg = [self.delegate entityForName:@"Message"];
    msg.source = buddy;
    [self.delegate.managedObjectContext refreshObject:buddy mergeChanges:YES];
    return msg;
}

- (void)asyncSave {
    [self.delegate save];
}

错误:

2012-08-10 00:28:20.526 聊天 [13170:c07] * -[AppDelegate 保存] 中的断言失败,/Users/paulp/Desktop/TestTask/Classes/AppDelegate.m:28 2012-08-10 00:28 :20.527 Chat[13170:c07] *由于未捕获的异常“NSInternalInconsistencyException”而终止应用程序,原因:“保存失败 {type = immutable dict, count = 2, entries => 1 : {contents = "NSAffectedObjectsErrorKey"} = (" (实体:句子;id:0x6b8bf10;数据:)") 2:{contents = "NSUnderlyingException"} = CoreData 无法满足 '0x6b8bf10' 的错误}

我究竟做错了什么?

更新 我指出了这一行的错误:

NSArray *sentenceArray = [self.context executeFetchRequest:request error:&error];

当我执行该行时,我得到了错误......那是在获取数据时。但是,将新数据保存到 Messages 实体时似乎出现了错误。随机句子是从句子中获取的。

在我将 asyncSave 方法更改为直接保存(因此不使用新线程)后,它会保存第一次聊天,但之后什么也没有。它死了。

更新 这一切似乎都在我的didFinishLaunchingWithOptions

[self.context setRetainsRegisteredObjects:YES];

我了解 CodeData 对象模型上下文不会释放其对象,这似乎是添加和保存之间的问题。但为什么?

4

5 回答 5

66

唔。您是否按照本指南正确实施并发?您看到的问题是跨多个线程使用核心数据时的常见问题。对象在您的“背景上下文”中被删除,而另一个上下文正在访问它。在删除之后但在保存之前调用[context processPendingChanges]您的背景上下文可能会有所帮助。

还有一个关于优化核心数据性能的 WWDC 2010 会议 (137),其中涉及到一些删除。

当您执行 fetch 时,Core Data 返回与您提供的谓词匹配的对象集合。这些对象实际上还没有设置它们的属性值。当您访问一个属性时,Core Data 会返回存储区以“触发故障” - 使用存储区中的数据填充该属性。“Could not fulfill a fault...”异常发生在 Core Data 去存储区获取对象的属性值,但该对象在持久存储区中不存在时。托管对象上下文认为它应该存在,这就是它可以尝试错误的原因——这就是问题所在。导致抛出异常的上下文不知道该对象已被其他东西(如另一个上下文)从存储中删除。

请注意,上面的并发指南现在已经过时了,您应该使用父子上下文和私有队列并发,而不是旧的线程限制模型。由于许多原因,父子上下文不太可能遇到“无法完成错误......”。请提交文档错误或使用反馈表来请求更新并发指南。

于 2012-08-15T23:07:13.670 回答
6

在不同的行中“保存”核心数据对象在概念上是不可能的。请记住,Core Data 是一个对象图而不是数据库。

如果您想“重新定位”您的句子,最好的方法是销毁它并重新创建它。如果要保留旧实例,只需创建一个新实例,然后从现有实例中填写属性。

为了破坏,使用

[self.context deleteObject:sentenceObject];

为了重新创建,使用

Sentence *newSentence = [NSEntityDescription insertNewObjectForEntityForName:
  "Sentences" inManagedObjectContext:self.context];
newSentence.sentence = sentenceObject.sentence;
// fill in other properties, then
[self.context save:error];

如果您想阅读此内容,请查看“Core Data Programming Guide”的“Using Managed Objects”部分中的“ Copying and Copy and Paste ”。

于 2012-08-13T13:12:56.167 回答
1

检查核心数据机制。“故障减少了您的应用程序消耗的内存量。故障是表示尚未完全实现的托管对象的占位符对象,或表示关系的集合对象:”

于 2012-10-30T03:39:55.597 回答
0

发生这种情况是因为您在完成获取第一次调用的所有关系之前将“随机消息”添加到新行。

您可以在第一次调用中添加预取以避免延迟加载,我相信问题将得到解决。

这是我们可以为请求进行预取的方法:

[request setRelationshipKeyPathsForPrefetching:[NSArray arrayWithObjects:@"whatEverOfYourWillNumberOne",@"whatEverOfYourWillNumberTwo", nil]];

希望有帮助。

于 2013-02-25T04:23:35.857 回答
0

我通过将 NSFetchedResultsController 的“cacheName”字符串更改为 nil 来修复错误。

NSFetchedResultsController *aFetchedResultsController = [[NSFetchedResultsController alloc] initWithFetchRequest:fetchRequest managedObjectContext:self.managedObjectContext sectionNameKeyPath:nil cacheName: @"Root" nil ];

于 2014-04-02T08:20:25.010 回答