当使用带有流式布局的UICollectionView的单元格滚动UITableView时,应用程序崩溃。

5

这个应用程序是使用Xcode 7.3.1编写的Swift 2.2。

我有一个带有UICollectionView和水平FlowLayout的单元格的UITableView。当我在设备(iPad Air 2,iOS 9.3.2)或模拟器上快速上下滚动表视图并且不让它停止时,应用程序会崩溃并显示以下错误信息:

 EXC_BAD_ACCESS

在AppDelegate中。

应用程序没有触发在视图控制器中设置的didReceiveMemoryWarning()断点。

看起来代码在以下位置崩溃:

[UICollectionViewData invalidateItemsAtIndexPaths:]

或者
[_UIFlowLayoutSection computeLayoutInRect:forSection:invalidating:]

跟随着

0_dispatch_barrier_async_f_slow.

堆栈跟踪没有指向任何应用程序代码的行。

模拟器的控制台堆栈跟踪(在设备崩溃时没有控制台堆栈跟踪):

*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '+[_UIFlowLayoutItem frame]: unrecognized selector sent to class 0x110f9ecc0'
*** First throw call stack:
(
0   CoreFoundation                      0x000000011259cd85 __exceptionPreprocess + 165
1   libobjc.A.dylib                     0x00000001137d3deb objc_exception_throw + 48
2   CoreFoundation                      0x00000001125a5c3d +[NSObject(NSObject) doesNotRecognizeSelector:] + 205
3   CoreFoundation                      0x00000001124ebcfa ___forwarding___ + 970
4   CoreFoundation                      0x000000011259e1f8 __forwarding_prep_1___ + 120
5   UIKit                               0x0000000110b1d8b5 -[UICollectionViewData invalidateItemsAtIndexPaths:] + 273
6   UIKit                               0x0000000110ae26b7 -[UICollectionView _invalidateWithBlock:] + 62
7   UIKit                               0x0000000110ae2a6f -[UICollectionView _invalidateLayoutWithContext:] + 656
8   UIKit                               0x0000000110af0ad9 -[UICollectionViewLayout invalidateLayoutWithContext:] + 189
9   UIKit                               0x0000000110afe2cc -[UICollectionViewFlowLayout invalidateLayoutWithContext:] + 604
10  libdispatch.dylib                   0x0000000114257d9d _dispatch_call_block_and_release + 12
11  libdispatch.dylib                   0x00000001142783eb _dispatch_client_callout + 8
12  libdispatch.dylib                   0x00000001142601ef _dispatch_main_queue_callback_4CF + 1738
13  CoreFoundation                      0x00000001124f60f9 __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__ + 9
14  CoreFoundation                      0x00000001124b7b99 __CFRunLoopRun + 2073
15  CoreFoundation                      0x00000001124b70f8 CFRunLoopRunSpecific + 488
16  GraphicsServices                    0x0000000114bcaad2 GSEventRunModal + 161
17  UIKit                               0x000000011024cf09 UIApplicationMain + 171
18  market-touch                        0x000000010d4cc472 main + 114
19  libdyld.dylib                       0x00000001142ac92d start + 1
20  ???                                 0x0000000000000001 0x0 + 1
)
libc++abi.dylib: terminating with uncaught exception of type NSException

这里是堆栈跟踪(稍微不同版本的这里)。
为什么会发生这种情况,有没有什么我可以做来防止这种情况发生?
编辑: 根据Doro的要求,这是我的cellForRowAtIndexPath:
func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {

    let cell = tableView.dequeueReusableCellWithIdentifier(customCellNibName, forIndexPath: indexPath) as! CustomTableViewCell

    //populating cell with data

    return cell 
}

我根本没有调用func setCollectionViewLayout(_ layout: UICollectionViewLayout, animated animated: Bool)

流式布局设置是在IB中完成的。

启用僵尸对象并没有对我有太大帮助,如果有的话。这是我得到的堆栈跟踪:

-[NSIndexPath frame]: unrecognized selector sent to instance 0xc000000012de4890
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[NSIndexPath frame]: unrecognized selector sent to instance 0xc000000012de4890'
*** First throw call stack:
(0x1815f6db0 0x180c5bf80 0x1815fdc4c 0x1815fabec 0x1814f8c5c 0x186fcccc0 0x186f9e688 0x1868de558 0x1868da0a4 0x1869ae9ec 0x1024f9a7c 0x1024f9a3c 0x1024ff4e4 0x1815acd50 0x1815aabb8 0x1814d4c50 0x182dbc088 0x1867b6088 0x1000c9f58 0x1810728b8)
libc++abi.dylib: terminating with uncaught exception of type NSException

编辑2:

在我的CustomTableViewCellawakeFromNib()方法中,我有以下代码:

if let layout = collectionView.collectionViewLayout as? UICollectionViewFlowLayout {
    layout.estimatedItemSize = CGSizeMake(50, 22)
}

我注意到,如果我注释掉这段代码,就不会再出现崩溃(尽管有时整个表视图会冻结且无法使用)。然而,此时集合视图单元格的大小不正确。请参考:正确 vs 错误
在IB中,集合视图流布局->项目大小为50x22。用于集合视图的单元格中有一个按钮,高度约束为22pt。以下是设置集合视图单元格布局属性的方法:
override func preferredLayoutAttributesFittingAttributes(layoutAttributes: UICollectionViewLayoutAttributes) -> UICollectionViewLayoutAttributes {
    let attributes = super.preferredLayoutAttributesFittingAttributes(layoutAttributes)

    button.setTitle(symbol, forState: .Normal)
    button.sizeToFit()

    attributes.frame = button.frame

    return attributes
}

这个想法是让这些单元格的高度为22pt,宽度不同(大约50pt),并且它们之间的距离相同。如果我将防止崩溃的代码注释掉,是否可以实现这一点?


你能分享一下你的cellForRowAtIndexPath:方法吗?这样会更好地理解。 - Doro
可能与https://dev59.com/HVkR5IYBdhLWcg3w2wp1相关的编程内容。仅返回翻译后的文本。 - Muhammad Ali
1个回答

1

请查看this教程。

基本上,这可能是由于UITableViewCell的可重用特性导致的。当您滚动tableView时,它使用相同的一组可见单元格,并从您的cellForRowAtIndexPath:方法设置内容。

根据您的实现方式,似乎在特定时间点UICollectionViewFlowLayout被释放。我的意思是,您调用了

func setCollectionViewLayout(_ layout: UICollectionViewLayout,
                    animated animated: Bool)

cellForRowAtIndexPath:方法内,当你滚动时,新的UICollectionViewLayout被分配并设置到了单元格内的collectionView中。在“编辑方案”-“诊断”-“启用僵尸对象”中启用僵尸对象。检查当前状态的正确性。
很难建议任何有效的代码片段,但作为一个想法,尝试在uitableviewCell的init方法中(仅一次)设置与流布局相关的所有属性,这将避免重新分配。
-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *CellIdentifier = @"CellIdentifier";

    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];

    if (!cell)
    {
        cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
        // set your flow layout or other stuff here
    }


    return cell;
}

希望这有所帮助。

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