UICollectionView大BUG — reloadData后变空白,cellForItemAtIndexPath不执行

最近项目中遇到的一个大坑,先摆出解决方案:https://github.com/tbl00c/UICollectionView-EmptyFix

症状

UICollectionView在reloadData后变成空白的了,contentSize正常,页面能够正常滑动,但是所有的cell都不展示了,visableCells属性为空。调试发现唯独cellForItemAtIndexPath这个回调方法不执行。


UICollectionView大BUG — reloadData后变空白,cellForItemAtIndexPath不执行_第1张图片
2.png

ps. 也有人遇到说界面上所有的cell的hidden属性都变为YES了,其实所有滑出页进入复用池的cell都会被hidden。

调研

那问题出在哪呢?
通过对比发现collectionView变空白前后的属性,发现_reloadingSuspendedCount这个私有属性的值被置为-1。


UICollectionView大BUG — reloadData后变空白,cellForItemAtIndexPath不执行_第2张图片
0.png

最终发现这个属性不为0时,reloadData是不会真正执行的,它在insert、delete、move操作前会先被++,以保证在执行相关动效是不被reloadData打乱,执行完毕后--。讲道理它不会变为负值,然而,它偏偏变了。

出错根源

_reloadingSuspendedCount为什么变为负值了呢?加入UICollectionView-EmptyFix,在它变为-1的时查看堆栈信息发现,这原来是另一个大坑:某个cell中,使用NSAttributedString处理html数据了,具体可以google。
就这样~。

解决方案

利用runtime将_resumeReloads替换,发现_reloadingSuspendedCount < 0时及时纠正。

参考

私有API调研过程参考:https://github.com/steipete/PSTCollectionView/tree/master/PSTCollectionView

你可能感兴趣的:(UICollectionView大BUG — reloadData后变空白,cellForItemAtIndexPath不执行)