App Store に最初のアプリを持っていますが、どのデバイスでも再現できないエラーが表示されます。あるユーザーは、アプリがクラッシュせずに製品を削除できないと報告しました。再現するために考えられることはすべて試しましたが、うまくいきませんでした。iTunes Connect でクラッシュの報告がなかったので、Crittercism にサインアップしました。これで、問題を抱えているユーザーが 3 人いることがわかりました。
問題は、クラッシュ ログを調べても、アプリがクラッシュする理由がわからないことです。アイテムを削除しても、クラッシュすることはありません。ユーザー入力情報のない新しく作成されたアイテムであってもです。問題を絞り込むのに役立つ可能性のある推奨事項は、大歓迎です。アプリはコア データを使用し、製品エンティティと他のエンティティのすべての削除ルールは、無効化またはカスケードに設定されています。
クラッシュ ログは、クラッシュの理由が「NSInvalidArgumentException - deleteObject: requires a non-nil argument」であることを示しています。バックトレースは次のとおりです。
Reported Name Reason App Version Symbolicated
15-May-12 07:22:06 PM NSInvalidArgumentException
-deleteObject: requires a non-nil argument
0 CoreFoundation 0x317a888f __exceptionPreprocess + 162
1 libobjc.A.dylib 0x315f3259 objc_exception_throw + 32
2 CoreData 0x3325cf23 -[NSManagedObjectContext deleteObject:] + 386
3 OnShelf 0x000972e7 0x76000 + 135911
4 CoreFoundation 0x317023fd -[NSObject performSelector:withObject:withObject:] + 52
5 UIKit 0x319f0e07 -[UIApplication sendAction:to:from:forEvent:] + 62
6 UIKit 0x31ab65e7 -[UIBarButtonItem(UIInternal) _sendAction:withEvent:] + 118
7 CoreFoundation 0x317023fd -[NSObject performSelector:withObject:withObject:] + 52
8 UIKit 0x319f0e07 -[UIApplication sendAction:to:from:forEvent:] + 62
9 UIKit 0x319f0dc3 -[UIApplication sendAction:toTarget:fromSender:forEvent:] + 30
10 UIKit 0x319f0da1 -[UIControl sendAction:to:forEvent:] + 44
11 UIKit 0x319f0b11 -[UIControl(Internal) _sendActionsForEvents:withEvent:] + 492
12 UIKit 0x319f1449 -[UIControl touchesEnded:withEvent:] + 476
13 UIKit 0x319ef92b -[UIWindow _sendTouchesForEvent:] + 318
14 UIKit 0x319ef319 -[UIWindow sendEvent:] + 380
15 UIKit 0x319d5695 -[UIApplication sendEvent:] + 356
16 UIKit 0x319d4f3b _UIApplicationHandleEvent + 5826
17 GraphicsServices 0x32bd822b PurpleEventCallback + 882
18 CoreFoundation 0x3177c523 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE1_PERFORM_FUNCTION__ + 38
19 CoreFoundation 0x3177c4c5 __CFRunLoopDoSource1 + 140
20 CoreFoundation 0x3177b313 __CFRunLoopRun + 1370
21 CoreFoundation 0x316fe4a5 CFRunLoopRunSpecific + 300
22 CoreFoundation 0x316fe36d CFRunLoopRunInMode + 104
23 GraphicsServices 0x32bd7439 GSEventRunModal + 136
24 UIKit 0x31a03cd5 UIApplicationMain + 1080
25 OnShelf 0x00077207 0x76000 + 4615
26 OnShelf 0x000771bc 0x76000 + 4540
UITableView で製品を削除するコードの唯一の異常な点は、最初に在庫数がゼロかどうかを確認することです。そうでない場合は、アラートが表示され、本当に製品を削除するかどうかをユーザーに尋ねます。私に連絡を取ったあるユーザーは、在庫数がゼロかどうかに関係なくクラッシュしたと言っていたので、クラッシュは表示されているアラートとは関係がないようです。コードは次のとおりです。
- (void)tableView:(UITableView *)tableView
commitEditingStyle:(UITableViewCellEditingStyle)editingStyle
forRowAtIndexPath:(NSIndexPath *)indexPath
{
Product *product = [fetchedResultsController objectAtIndexPath:indexPath];
// Alert the user if the item is currently in stock
if ([product.details.quantity intValue] > 0) {
[self showAlert];
if (editingStyle == UITableViewCellEditingStyleDelete) {
lastIndexPath = indexPath;
toDelete = YES;
NSLog(@"Yes delete");
} else {
toDelete = NO;
}
} else {
// Delete the managed object.
NSManagedObjectContext *myContext = [fetchedResultsController managedObjectContext];
[myContext deleteObject:[fetchedResultsController objectAtIndexPath:indexPath]];
NSError *error;
if (![myContext save:&error]) {
// Update to handle the error appropriately.
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
exit(-1); // Fail
}
}
}
#pragma mark - AlertView Delegate Methods
- (void)showAlert {
// open a alert with an OK and cancel button
UIAlertView *alert =[[UIAlertView alloc] initWithTitle: @"Product In Stock"
message: @"All product data will be deleted"
delegate: nil
cancelButtonTitle: @"Cancel"
otherButtonTitles:@"OK", nil];
[alert setDelegate: self];
[alert show];
}
- (void)alertView:(UIAlertView *)alertView clickedButtonAtIndex:(NSInteger)buttonIndex{
if (buttonIndex == 0) {
//do nothing
}
else
{
if( toDelete == YES)
{
// Delete the managed object.
NSManagedObjectContext *myContext = [fetchedResultsController managedObjectContext];
[myContext deleteObject:[fetchedResultsController objectAtIndexPath: lastIndexPath]];
NSError *error;
if (![myContext save:&error]) {
// Update to handle the error appropriately.
NSLog(@"Unresolved error %@, %@", error, [error userInfo]);
exit(-1); // Fail
}
}
}
}
Apple チュートリアルのボイラープレート NSFetchedResultsController デリゲート メソッドを使用しています。
- (void)controllerWillChangeContent:(NSFetchedResultsController *)controller {
// The fetch controller is about to start sending change notifications, so prepare the table view for updates.
[self.tableView beginUpdates];
}
- (void)controller:(NSFetchedResultsController *)controller didChangeObject:(id)anObject
atIndexPath:(NSIndexPath *)indexPath
forChangeType:(NSFetchedResultsChangeType)type
newIndexPath:(NSIndexPath *)newIndexPath {
UITableView *aTableView = self.tableView;
switch(type) {
case NSFetchedResultsChangeInsert:
[aTableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
break;
case NSFetchedResultsChangeDelete:
[aTableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
break;
case NSFetchedResultsChangeUpdate:
[self configureCell:(DatabaseCell *)[aTableView cellForRowAtIndexPath:indexPath] atIndexPath:indexPath];
break;
case NSFetchedResultsChangeMove:
[aTableView deleteRowsAtIndexPaths:[NSArray arrayWithObject:indexPath] withRowAnimation:UITableViewRowAnimationFade];
[aTableView insertRowsAtIndexPaths:[NSArray arrayWithObject:newIndexPath] withRowAnimation:UITableViewRowAnimationFade];
break;
}
}
- (void)controller:(NSFetchedResultsController *)controller didChangeSection:(id <NSFetchedResultsSectionInfo>)sectionInfo
atIndex:(NSUInteger)sectionIndex
forChangeType:(NSFetchedResultsChangeType)type {
switch(type) {
case NSFetchedResultsChangeInsert:
[self.tableView insertSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
break;
case NSFetchedResultsChangeDelete:
[self.tableView deleteSections:[NSIndexSet indexSetWithIndex:sectionIndex] withRowAnimation:UITableViewRowAnimationFade];
break;
}
}
- (void)controllerDidChangeContent:(NSFetchedResultsController *)controller {
// The fetch controller has sent all current change notifications, so tell the table view to process all updates.
[self.tableView endUpdates];
}