iOS 7下UITableView section index相关崩溃

Posted

技术标签:

【中文标题】iOS 7下UITableView section index相关崩溃【英文标题】:UITableView section index related crashes under iOS 7 【发布时间】:2013-09-20 14:17:04 【问题描述】:

我有一个简单的表视图部分索引,它第一次看起来很好,但如果发生某些强制重绘部分索引的交互,则会崩溃,方法是:

    通过点击其中一个单元格来推送视图控制器 点击表格视图标题中的搜索栏或通过其他方式使其成为第一响应者 转到信息屏幕,然后返回带有该表视图的视图控制器。

如果我注释掉节索引(通过在 sectionIndexTitlesForTableView 中返回 nil 而不是“[self.collat​​ion sectionIndexTitles]”),它就可以正常工作。

它只发生在 ios 7 下(因为测试版并且在 7.0.1 版本中没有修复),我总是得到以下与 CoreText/动画/过渡相关的堆栈跟踪:

Exception Type:  EXC_BAD_ACCESS (SIGSEGV)
Exception Subtype: KERN_INVALID_ADDRESS at 0x000001f0
Triggered by Thread:  0

Thread 0 Crashed:
0   CoreText                        0x31403312 TComponentFont::GetMinSideBearing(CGAffineTransform const&, bool) const + 10
1   CoreText                        0x313f3ada TRun::GetLeftHangersGlyphCountAndWidth(long, unsigned long, TCharStream const&) const + 322
2   CoreText                        0x313f394c TLine::GetLeftHangersGlyphCountAndWidth(unsigned long const*) const + 148
3   CoreText                        0x314025d8 TLine::GetBounds(unsigned long) const + 496
4   CoreText                        0x314023ca CTLineGetBoundsWithOptions + 10
5   UIKit                           0x333641f6 -[UITableViewIndex _cacheAndMeasureTitles] + 758
6   UIKit                           0x33364b8e -[UITableViewIndex sizeThatFits:] + 198
7   UIKit                           0x332f0d62 -[UITableView _updateIndexFrameSuppressingChangeNotification:] + 546
8   UIKit                           0x332f6fe0 -[UITableView setContentInset:] + 268
9   UIKit                           0x332cbaf8 -[UIViewController _setNavigationControllerContentInsetAdjustment:] + 436
10  UIKit                           0x332e432e -[UINavigationController navigationTransitionView:didEndTransition:fromView:toView:] + 458
11  UIKit                           0x333af438 __49-[UINavigationController _startCustomTransition:]_block_invoke + 156
12  UIKit                           0x333340b0 -[_UIViewControllerTransitionContext completeTransition:] + 72
13  UIKit                           0x333f4402 __53-[_UINavigationParallaxTransition animateTransition:]_block_invoke105 + 450
14  UIKit                           0x332557c6 -[UIViewAnimationBlockDelegate _didEndBlockAnimation:finished:context:] + 282
15  UIKit                           0x33255412 -[UIViewAnimationState sendDelegateAnimationDidStop:finished:] + 174
16  UIKit                           0x3325532a -[UIViewAnimationState animationDidStop:finished:] + 66
17  QuartzCore                      0x32eadd94 CA::Layer::run_animation_callbacks(void*) + 228
18  libdispatch.dylib               0x3b21cd64 _dispatch_client_callout + 20
19  libdispatch.dylib               0x3b2237bc _dispatch_main_queue_callback_4CF$VARIANT$mp + 264
20  CoreFoundation                  0x30d1480c __CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__ + 4
21  CoreFoundation                  0x30d130e0 __CFRunLoopRun + 1296
22  CoreFoundation                  0x30c7dcd2 CFRunLoopRunSpecific + 518
23  CoreFoundation                  0x30c7dab6 CFRunLoopRunInMode + 102
24  GraphicsServices                0x356c32d6 GSEventRunModal + 134
25  UIKit                           0x3328211c UIApplicationMain + 1132

我几乎排除了它与内存有关,因为它发生在一个空表以及 6000 个单元格中。

这是节索引的代码:

- (NSString *)tableView:(UITableView *)tableView titleForHeaderInSection:(NSInteger)section 

    // Don't show section titles when a search is in progress
    if (isFilterInProgress && [self.searchDisplayController isActive]) 
        return nil;
    

    return  [[self.collation sectionIndexTitles] objectAtIndex:section];



- (NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView 

    if(tableView == self.resultsTable)

        // Don't show section titles when a search is in progress
        if (isFilterInProgress && [self.searchDisplayController isActive]) 
            return nil;
        

        // Don't show section titles when there are filtered results
        if ([self.filteredListContent count] > 0)
            return nil;

        // iOS7 temporary fix for crashes (ICM-9)
        if(SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"7.0"))

            // Crashes under iOS 7 when displayed for a second time!
            return [self.collation sectionIndexTitles];
            //return nil;

        else

            return [self.collation sectionIndexTitles];  

        

    else return nil;



- (NSInteger)tableView:(UITableView *)tableView sectionForSectionIndexTitle:(NSString *)title atIndex:(NSInteger)index 


    // Return correct section
    if(tableView == self.resultsTable)
        return [self.collation sectionForSectionIndexTitleAtIndex:index];
    else
        return 0;
       

还有其他人有类似的问题或与部分索引相关的任何其他问题吗?我知道表格视图部分索引以及编辑单元格上的删除按钮存在一些布局问题,但我不确定这些是否已修复。我怀疑这只是 iOS 7 中的一个错误?

【问题讨论】:

【参考方案1】:

我一直遇到这个确切的问题。经过几天的挫折后,我通过使用第 3 方部分索引视图而不使用本机 sectionIndexTitlesForTableView 解决了这个问题。查看CMIndexBar。超级简单,大概5分钟就搞定了。轻松定制。在 Apple 解决此错误之前,我将一直使用它,如果您遇到此崩溃,我强烈推荐它。

【讨论】:

Apple 确认这是框架中的一个错误,没有说明何时修复。自行决定是否采取临时解决办法。 Aaa 他们退后说这是我的代码中的一个错误。其他人可以发布他们发生这种情况的具体情况吗?对我来说,它似乎是 tableView 与标题中的搜索栏的组合,而 tableView 是 ViewController 本身的视图。仅当性索引重新出现时才会发生崩溃。 对我来说也可以重现,尽管在另一个 ViewController 出现并且 UITableView(带有搜索栏和部分索引)在导航中被推到左侧之后发生崩溃。我们是否有可能以不同的方式复制“相同的错误”? 马克斯,我不确定。如果你是异步加载数据,你是如何重新加载表的?【参考方案2】:

在我的情况下,这个问题似乎与从后台线程上的后台线程尝试的 reloadData 调用有关。我猜视图中的数据与索引或类似的东西不一致。

奇怪的是,它以 CoreText 库(如上面的堆栈跟踪中所见)和仅在 iOS 7 中看到的形式表现出来。

在主队列上调度调用以重新加载表视图的数据时,重新加载性能得到改善,我可以再次使用节索引。从那以后我再也没有发生过那次崩溃。

看看这样做是否能解决你的问题。

【讨论】:

我们遇到了与 UIAlertView 或 UIActionSheet 相同的问题。在此之后任何加载表的尝试都会导致崩溃。我试图在主队列上运行它,但没有成功。你确定这行得通吗? 7.1 beta 3 运行相同的代码而没有崩溃。 micap,至少它为我解决了这个问题,重要的部分是在主队列上运行“reloadData”(以及警报演示,因为它是一个与 UI 相关的任务,我猜),是你确定你正在这样做吗?您确定在点击警报按钮后或在您完成选择器后没有后台线程实际上触发表重新加载吗? 我在主线程上调用 reloadData,但仍然有这个问题 com.apple.main-thread EXC_BAD_ACCESS KERN_INVALID_ADDRESS TComponentFont::GetMinSideBearing(CGAffineTransform const&, bool) const TRun::GetLeftHangersGlyphCountAndWidth(long, unsigned long, TCharStream const&) const TLine::GetLeftHangersGlyphCountAndWidth(unsigned long const*) const TLine::GetBounds(unsigned long) const CTLineGetBoundsWithOptions -[UITableViewIndex _cacheAndMeasureTitles] -[UITableViewIndex setTitles:] + 134 -[UITableView _updateIndex] + 548 -[UITableView noteNumberOfRowsChanged] + 170 -[UITableView reloadData] + 796【参考方案3】:

我发现问题出在函数中:

- (NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView

iOS 7.0 以后的节索引算法有问题,很明显是框架的bug。我结束了这样的事情:

- (NSArray *)sectionIndexTitlesForTableView:(UITableView *)tableView  
        **if(IS_IOS7 || isSearching)
            return nil;**
        else
            return [self.dictAllKeysAlphabetically allKeys]; 

这是一个丑陋的补丁,它会从屏幕右侧移除字母栏,但至少你的应用不会崩溃。

【讨论】:

正如我在“如果我注释掉节索引(通过在 sectionIndexTitlesForTableView 中返回 nil 而不是“[self.collat​​ion sectionIndexTitles]”)中提到的,它工作得很好。”【参考方案4】:

它已在 7.1 测试版中得到修复。

【讨论】:

以上是关于iOS 7下UITableView section index相关崩溃的主要内容,如果未能解决你的问题,请参考以下文章

UITableView设置Section间距

iOS8中 UITableView section 分区头部视图不显示

IOS 7:从分组样式 UITableView 的一部分中删除边框

如何处理 iOS UITableView 中的动态 Sections 和 Rows

UISearchController:UITableView 的 Section Index 与 searchBar 重叠

iOS UITableView 滑动时顺序混乱或多个cell内容相同