NSFetchedResultsController Delete All
AcaniChat, like the native iPhone Messages app, has a method to clear all messages:
for (Message *message in [fetchedResultsController_ fetchedObjects]) {
[managedObjectContext_ deleteObject:message];
}
if (![managedObjectContext_ save:&error]) {
// TODO: Handle the error appropriately.
NSLog(@"Delete message error %@, %@", error, [error userInfo]);
}
But, only when there's a lot of messages being deleted (enough that some are off-screen), AcaniChat crashes with the errors:
2011-04-10 01:06:17.629 AcaniChat[11129:207] Serious application error.开发者_StackOverflow Exception was caught during Core Data change processing. This is usually a bug within an observer of NSManagedObjectContextObjectsDidChangeNotification. CoreData could not fulfill a fault for '0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42>' with userInfo {
NSAffectedObjectsErrorKey = (
"<Message: 0x4d665b0> (entity: Message; id: 0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42> ; data: <fault>)"
);
}
2011-04-10 01:06:17.721 AcaniChat[11129:207] Delete message error Error Domain=NSCocoaErrorDomain Code=133000 "The operation couldn’t be completed. (Cocoa error 133000.)" UserInfo=0x4d9cf70 {NSAffectedObjectsErrorKey=(
"<Message: 0x4d665b0> (entity: Message; id: 0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42> ; data: <fault>)"
), NSUnderlyingException=CoreData could not fulfill a fault for '0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42>'}, {
NSAffectedObjectsErrorKey = (
"<Message: 0x4d665b0> (entity: Message; id: 0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42> ; data: <fault>)"
);
NSUnderlyingException = "CoreData could not fulfill a fault for '0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42>'";
}
2011-04-10 01:06:18.036 AcaniChat[11129:207] CoreAnimation: ignoring exception: CoreData could not fulfill a fault for '0x4d7d050 <x-coredata://F07CAEF9-5F38-4A1F-BA17-52A3E2E56978/Message/p42>'
How do I fix this?
I fixed this by setting the fetchedResultsController_.delegate = nil
, deleting all objects in cellMap
(which is actually the data source for the UITableView *chatContent
because I add NSDate
objects to cellMap
for the timestamp cells), and then, setting the fetchedResultsController_.delegate = self
and performing the fetch request again.
NSError *error;
fetchedResultsController.delegate = nil; // turn off delegate callbacks
for (Message *message in [fetchedResultsController fetchedObjects]) {
[managedObjectContext deleteObject:message];
}
if (![managedObjectContext save:&error]) {
// TODO: Handle the error appropriately.
NSLog(@"Delete message error %@, %@", error, [error userInfo]);
}
fetchedResultsController.delegate = self; // reconnect after mass delete
if (![fetchedResultsController performFetch:&error]) { // resync controller
// TODO: Handle the error appropriately.
NSLog(@"fetchMessages error %@, %@", error, [error userInfo]);
}
[cellMap removeAllObjects];
[chatContent reloadData];
Also, check out MagicalRecord. It has a +truncateAll
method.
精彩评论