应用程序由于未捕获的异常“NSInternalInconsistencyException”而终止。

4
如果用户触摸UITableView并只更新以下代码中的TableView
[self.myTableView beginUpdates];
[myTableView endUpdates];

然后它会产生以下崩溃报告。
2013-12-17 17:27:33.446 planobot[12300:a0b] *** Assertion failure in -[UITableView _endCellAnimationsWithContext:], /SourceCache/UIKit_Sim/UIKit-2903.23/UITableView.m:1330
2013-12-17 17:27:33.469 planobot[12300:a0b] *** Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Invalid update: invalid number of rows in section 0.  The number of rows contained in an existing section after the update (1) must be equal to the number of rows contained in that section before the update (7), plus or minus the number of rows inserted or deleted from that section (0 inserted, 0 deleted) and plus or minus the number of rows moved into or out of that section (0 moved in, 0 moved out).'
*** First throw call stack:
(
    0   CoreFoundation                      0x0210e5e4 __exceptionPreprocess + 180
    1   libobjc.A.dylib                     0x01e918b6 objc_exception_throw + 44
    2   CoreFoundation                      0x0210e448 +[NSException raise:format:arguments:] + 136
    3   Foundation                          0x00fe5fee -[NSAssertionHandler handleFailureInMethod:object:file:lineNumber:description:] + 116
    4   UIKit                               0x0024485d -[UITableView _endCellAnimationsWithContext:] + 13402
    5   UIKit                               0x00253caa -[UITableView endUpdatesWithContext:] + 51
    6   UIKit                               0x00253cd8 -[UITableView endUpdates] + 41
    7   planobot                            0x0004f34a -[DailyReportViewController tableView:didSelectRowAtIndexPath:] + 1658
    8   UIKit                               0x002557b1 -[UITableView _selectRowAtIndexPath:animated:scrollPosition:notifyDelegate:] + 1513
    9   UIKit                               0x00255924 -[UITableView _userSelectRowAtPendingSelectionIndexPath:] + 279
    10  UIKit                               0x00259908 __38-[UITableView touchesEnded:withEvent:]_block_invoke + 43
    11  UIKit                               0x00190183 ___afterCACommitHandler_block_invoke + 15
    12  UIKit                               0x0019012e _applyBlockToCFArrayCopiedToStack + 403
    13  UIKit                               0x0018ff5a _afterCACommitHandler + 532
    14  CoreFoundation                      0x020d64ce __CFRUNLOOP_IS_CALLING_OUT_TO_AN_OBSERVER_CALLBACK_FUNCTION__ + 30
    15  CoreFoundation                      0x020d641f __CFRunLoopDoObservers + 399
    16  CoreFoundation                      0x020b4344 __CFRunLoopRun + 1076
    17  CoreFoundation                      0x020b3ac3 CFRunLoopRunSpecific + 467
    18  CoreFoundation                      0x020b38db CFRunLoopRunInMode + 123
    19  GraphicsServices                    0x02fec9e2 GSEventRunModal + 192
    20  GraphicsServices                    0x02fec809 GSEventRun + 104
    21  UIKit                               0x00173d3b UIApplicationMain + 1225
    22  planobot                            0x000a6a4d main + 141
    23  planobot                            0x000022b5 start + 53
)
libc++abi.dylib: terminating with uncaught exception of type NSException

为什么应用程序会抛出NSInternalInconsistencyException异常?

在 beginUpdates 和 endUpdates 方法块之间你正在更新什么内容?为什么需要调用这两个方法? - Suryakant Sharma
1
我对不同的表格视图引用很好奇。从 NSLog(@"Property %@, Variable %@", self.myTableView, myTableView); 中你能看到什么? - Phillip Mills
3
你的表格视图数据源在上次重新加载后发生了变化。上次重新加载时有7个数据元素,而现在(当你可以开始/结束更新)数据源数组中只有一个数据元素。请确保在修改数据源时更新表格视图。 - Putz1103
嗨,我经常收到这个错误并解决了。请让我看看整个代码。 - Matteo Gobbi
2个回答

1

而不是这样:

[self.myTableView beginUpdates];
[myTableView endUpdates];

try this:

[self.myTableView beginUpdates];
[self.myTableView endUpdates];

我怀疑你的myTableView属性没有调用endUpdates。然而,已经调用了beginUpdates,这导致了以下问题:
  1. beginUpdates/endUpdates调用不平衡
  2. NSInternalInconsistencyException

1

当我的计数不正确时,我经常会看到这个。根据此:

reason: 'Invalid update: invalid number of rows in section 0.  The number of rows contained in an existing section after the update (1) must be equal to the number of rows contained in that section before the update (7), plus or minus the number of rows inserted or deleted from that section (0 inserted, 0 deleted) and plus or minus the number of rows moved into or out of that section (0 moved in, 0 moved out).'

我猜测您的表格中有7个元素(如tableView:numberOfRowsInSection:所报告的那样),现在返回1。这让我相信您认为调用了
 [self.myTableView beginUpdates];
 [myTableView endUpdates];

当您删除行时,这已经足够了。

但是您需要首先告诉UITableView哪些行被删除(或添加)。 这将取决于您的具体情况,但如果您想要删除第零节的前七行,代码如下。

NSMutableArray *indexes = [[NSMutableArray alloc] init];
for (int i = 0; i <= 6; i++)
{
    [indexes addObject:[NSIndexPath indexPathForItem:i inSection:0]];
}
[myTable beginUpdates];
[myTable deleteRowsAtIndexPaths:indexes withRowAnimation:NO];
[myTable endUpdates];

正如@Marco所说,只需使用指向表格的相同指针,否则会有些混淆,这是我在示例中所做的。

或者如果你只想让你的表格完全重新加载,你可以调用reloadData。

[myTable reloadData];

我认为这种方式的效率比手动指定添加或删除的行要低,但在一个包含7个表格的表格中,这种方式是可以接受的。

[myTable reloadData] 这个方法并不低效,根据它的文档所述:“为了提高效率,表视图只会重新显示可见的行。” - Diego Palomar

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接