EXC_BAD_ACCESS 长按;没有僵尸 activity

EXC_BAD_ACCESS on long press; no zombie activity

编辑:我已经确定 在 iPad Retina 模拟器和我的 iPad Mini 3 测试设备上发生以下情况。它在这些目标上 100% 的时间发生,而在任何其他 iPad 或 iPhone 模拟器上从未发生过(iPad Air/Air 2;iPad Pro;iPhone 5 到 7 加)。我没有任何其他测试设备。


我有一个用 Swift 编写的应用程序,它显示一个 UITableView,我在每个单元格内的自定义视图中添加了一个 UILongPressGestureRecognizer。该应用程序 运行 运行良好,直到我连续长按,此时应用程序停止并显示

Thread 1: EXC_BAD_ACCESS (code=1, address=0xd08600b6)

(地址变化很大,看起来很漂亮 运行dom。有时它是 0x0。)当崩溃发生时,调试控制台中没有输出。 XCode 本身打开到我的应用程序委托的 class 声明的第一行。

我在我的长按事件处理程序方法的第一行放置了一个断点,但应用程序在命中断点之前崩溃了。我也加了一个异常断点,但是也没抓到问题

我在方案中启用了僵尸对象,但没有任何显示。我已经尝试 运行 使用 Instruments 和僵尸模板来安装应用程序;该应用程序只是停止并且 Instruments 没有标记任何僵尸 activity。 XCode 的分析报告没有问题。

我不知道它有什么相关性,但这个问题是新出现的,因为我首先将应用程序存档以准备分发给测试人员。在此之前,该应用程序可以在模拟器和测试设备上完美运行。问题首先出现在我第一次安装和 运行 设备上的分发 .ipa 文件时。然后我回到模拟器并发现了相同的行为。我已经清理了项目,清理了构建文件夹,重新启动了我的开发机器,并在原始状态下完成了我能想到的一切 运行 。没有区别。

我不知道如何在跟踪这个问题上取得进展。任何建议将不胜感激。

更多信息

似乎崩溃发生在传递给 UILongPressGestureRecognizer 初始值设定项的 #selector 对象内部。这是调试导航器窗格中的调用堆栈:

#0 0x006e7295 in objc_retain ()
#1 0x0007529f in @objc TikkunRowView.onLongPress(sender : UILongPressGestureRecognizer, forEvent : UIEvent) -> () ()
#2 0x01cc6d3b in -[UIGestureRecognizerTarget _sendActionWithGestureRecognizer:] ()
#3 0x01ccf75e in _UIGestureRecognizerSendTargetActions ()
#4 0x01cccdee in _UIGestureRecognizerSendActions ()
#5 0x01ccbca8 in -[UIGestureRecognizer _updateGestureWithEvent:buttonEvent:] ()
#6 0x01cb6670 in _UIGestureEnvironmentUpdate ()

这里是上面 #1 中发生崩溃的代码(偏移量 <+42> 处的 calll):

Tikkun Sample`@objc TikkunRowView.onLongPress(sender : UILongPressGestureRecognizer, forEvent : UIEvent) -> ():
    0x60270 <+0>:   pushl  %ebp
    0x60271 <+1>:   movl   %esp, %ebp
    0x60273 <+3>:   pushl  %esi
    0x60274 <+4>:   subl   [=12=]x24, %esp
    0x60277 <+7>:   movl   0x14(%ebp), %eax
    0x6027a <+10>:  movl   0x10(%ebp), %ecx
    0x6027d <+13>:  movl   0x8(%ebp), %edx
    0x60280 <+16>:  movl   %ecx, (%esp)
    0x60283 <+19>:  movl   %ecx, -0x8(%ebp)
    0x60286 <+22>:  movl   %edx, -0xc(%ebp)
    0x60289 <+25>:  movl   %eax, -0x10(%ebp)
    0x6028c <+28>:  calll  0x7c9ec                   ; symbol stub for: objc_retain
    0x60291 <+33>:  movl   -0x10(%ebp), %ecx
    0x60294 <+36>:  movl   %ecx, (%esp)
    0x60297 <+39>:  movl   %eax, -0x14(%ebp)
    0x6029a <+42>:  calll  0x7c9ec                   ; symbol stub for: objc_retain
>>  0x6029f <+47>:  movl   -0xc(%ebp), %ecx << CRASH HAPPENS WITH THIS LINE HIGHLIGHTED
    0x602a2 <+50>:  movl   %ecx, (%esp)
    0x602a5 <+53>:  movl   %eax, -0x18(%ebp)
    0x602a8 <+56>:  calll  0x7c9ec                   ; symbol stub for: objc_retain
    0x602ad <+61>:  movl   -0x8(%ebp), %ecx
    0x602b0 <+64>:  movl   %ecx, (%esp)
    0x602b3 <+67>:  movl   -0x10(%ebp), %edx
    0x602b6 <+70>:  movl   %edx, 0x4(%esp)
    0x602ba <+74>:  movl   -0xc(%ebp), %esi
    0x602bd <+77>:  movl   %esi, 0x8(%esp)
    0x602c1 <+81>:  movl   %eax, -0x1c(%ebp)
    0x602c4 <+84>:  calll  0x5fd40                   ; Tikkun_Sample.TikkunRowView.onLongPress (sender : __ObjC.UILongPressGestureRecognizer, forEvent : __ObjC.UIEvent) -> () at TikkunRowView.swift:128
    0x602c9 <+89>:  movl   -0xc(%ebp), %eax
    0x602cc <+92>:  movl   %eax, (%esp)
    0x602cf <+95>:  calll  0x7c9e6                   ; symbol stub for: objc_release
    0x602d4 <+100>: addl   [=12=]x24, %esp
    0x602d7 <+103>: popl   %esi
    0x602d8 <+104>: popl   %ebp
    0x602d9 <+105>: retl   

我不是阅读这些东西的专家,但看起来崩溃发生在编译器生成的代码中,该代码应该分派到我的事件处理程序方法。我不知道这段代码在调用我的代码之前检查了哪些对象(在偏移 <+84> 处)。这是我用来构建手势识别器的代码:

let g = UILongPressGestureRecognizer(target: self, action: #selector(TikkunRowView.onLongPress(sender:forEvent:)))

我不知道为什么控制台没有给你任何详细的错误信息。 (它应该!)但是看看你的代码,这显然是由于手势动作回调的错误签名。以下内容引自 api 文档。

The action methods invoked must conform to one of the following signatures:
- (void)handleGesture;
- (void)handleGesture:(UIGestureRecognizer *)gestureRecognizer;