- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
有几个问题提出了与此完全相反的问题,我不明白如何/为什么在 Release模式下运行我的应用程序,但崩溃并显示 EXC_BAD_ACCESS
Debug模式错误。
崩溃的方法是递归的,而且极其!!重大的;只要没有太多递归,它在调试(在 iPhone XS 上少于 ~1000,在模拟器上无限制)和 Release模式(无限制?)都可以正常工作。
我不知道从哪里开始找到如何调试 Debug模式,我想知道是否由于堆栈跟踪或其他未知原因捆绑了某种递归软限制?甚至可以归结为电缆,因为我能够在模拟器中成功运行而没有问题吗?
我应该注意到,Xcode 会在看似随机的位置报告崩溃,例如我知道已实例化且有效的属性 getter;以防万一。
我打算将它重构为更小的 block ,但我想我会在这里发帖,以防有人对可能导致此问题的原因有任何想法。
参见: https://gist.github.com/ThomasHaz/3aa89cc9b7bda6d98618449c9d6ea1e1
最佳答案
您的堆栈内存不足。
考虑这个非常简单的递归函数,将 1 到 n 之间的整数相加:
func sum(to n: Int) -> Int {
guard n > 0 else { return 0 }
return n + sum(to: n - 1)
}
您会发现,例如,如果您尝试对 1 到 100,000 之间的数字求和,应用程序在发布和调试版本中都会崩溃,但在调试版本中只会更快崩溃。我怀疑在调试版本中只是将更多诊断信息推送到堆栈上,导致堆栈中的空间更快耗尽。在上面的发布版本中,堆栈指针每次递归调用前移 0x20 字节,而调试构建每次前移 0x80 字节。而且,如果您在递归函数中做任何实质性的事情,这些增量可能会更大,并且可能会在递归调用更少的情况下发生崩溃。但是我的设备(iPhone Xs Max)和我的模拟器(Thread.current.stackSize
)上的堆栈大小是 524,288 字节,这对应于堆栈指针前进的数量和我能够进行的递归调用的最大数量达到。如果您的设备比模拟器更早崩溃,则可能是您的设备的 RAM 较少,因此分配了较小的 stackSize
。
最重要的是,如果您想享受快速性能但又不想招致巨大调用堆栈的内存开销,则可能需要将您的算法重构为非递归算法。顺便说一句,上述的非递归再现比递归再现快一个数量级。
或者,您可以异步分派(dispatch)递归调用,这消除了堆栈大小问题,但引入了 GCD 开销。上面的异步再现比简单的递归再现慢两到三个数量级,显然,比迭代再现慢另一个数量级。
诚然,我的简单 sum
方法非常简单,递归调用的开销开始占总计算时间的很大一部分,并且假设您的例程看起来更复杂,我怀疑差异不会那么明显。尽管如此,如果您想避免用完堆栈空间,我只是建议采用非递归再现。
我建议您观看以下 WWDC 视频:
值得注意的是,深度递归例程并不总是需要消耗大量堆栈。值得注意的是,有时我们可以使用 tail-recursion我们的递归调用是最后一次调用。例如。我上面的代码片段没有使用尾调用,因为它将 n
添加到递归调用返回的值中。但我们可以重构它以传递运行总数,从而确保递归调用是真正的“尾调用”:
func sum(to n: Int, previousTotal: Int = 0) -> Int {
guard n > 0 else { return previousTotal }
return sum(to: n - 1, previousTotal: previousTotal + n)
}
发布版本足够智能来优化这种尾递归(通过称为“尾调用优化”的过程,TCO,也称为“尾调用消除”),从而减轻递归调用的堆栈增长。全局开发者大会 2015 Profiling in Depth ,而在另一个主题上,时间分析器准确地显示了优化尾调用时发生的情况。
最终效果是,如果您的递归例程使用尾调用,发布版本可以使用尾调用消除来缓解堆栈内存问题,但调试(未优化)版本不会这样做。
关于iOS 应用程序在 Debug模式下崩溃,在 Release模式下工作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55789071/
Debug.Assert/Debug.Fail 是否自动条件编译#if "DEBUG"?或者它是否更像是没有附加调试器(即使在发行版中)它什么也做不了?如果是这样,将它们留在您的代码中是否会对性能产生
我有一个应用程序,我配置了多个路由,一切正常,直到我配置的最新路由不起作用(显示错误的屏幕)。 我的问题是如何进行调试?没有打印错误日志,我无法找到如何获取有关正在发生的事情的更多日志。我也不知道从哪
我正在 Intellij 中调试代码。我使用 maven 来构建项目,并且在本地 .m2 存储库中有该项目的各种版本。当我开始调试时,Intellij 继续从项目的前一个快照中选择旧版本的代码。如何让
我喜欢在业余时间进行一些 TiVo 黑客事件 - TiVo 使用 Linux 变体和 TCL 。我想在我的 Windows 笔记本电脑上编写 TCL 脚本,测试它们,然后将它们通过 FTP 传输到我的
我有 ASM 代码,它使用循环语法打印 abc 。这是我的代码 ;abc.com .model small .code org 100h start: mov ah, 02h mov
我在 Debugging .net 2.0 Applications 中看到了以下代码 [Conditional("DEBUG")] void AssertTableExists() { #i
在大型项目中哪个更好用,为什么: #if DEBUG public void SetPrivateValue(int value) { ... } #endif 或 [System.D
我似乎无法让调试器运行。调试运行图标变灰,菜单选项丢失。 这只是main的情况,我可以很好地调试单元测试。 类似的问题提到了项目结构,但我看不出有什么不对: $GOPATH/src/foo.bar.c
只是想知道我的浏览器一直询问我是否想在每次点击浏览器链接刷新时停止调试非常烦人,因为这会减慢开发时间。 有没有其他人遇到过这个? 干杯 最佳答案 更新的答案,现在找到根本原因 经过两年看到这个错误时断
我正在尝试包含调试/发布相关编译器标志,例如: set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=gnu++0x -Wall -DUSE_BOOST") set
当我尝试使用 debug.phonegap.com 调试我的phonegap 应用程序时遇到问题。 我把这个视频放在 HTML 文档的头部 在启动应用程序之前,我从 build.phonegap.
GDB 7.0以后,支持反向调试。 生成核心转储时,我可以使用反向调试命令吗? 我怎样才能做到这一点? 最佳答案 你不能。核心文件是某个时间点程序状态的快照。要在该状态下向后移动,您需要程序状态的较早
首先:如果之前有人问过这个问题,我很抱歉。我是一个熟练的谷歌用户,但这确实让我难住了,我找不到任何东西。 我目前正在编写一个小型库,我想对其进行调试。我还希望能够完全关闭调试,并且编译后的代码不应包含
我想在 tomcat 中将级别日志记录设置为 DEBUG,但在控制台中仍然只有 INFO 和 WARN 输出。谁能告诉我哪里出了问题? 我的 C:\tomcat\logging.properties:
我已经开始像这样使用定义类了: internal sealed class Defines { /// /// This constant is set to true iff th
在使用编译器指令时,我不清楚以下两个代码片段中哪一个是正确/首选的,以及为什么。似乎我见过的大多数开发人员和开源项目都使用第一种,但我也看到第二种也经常使用。 #ifdef DEBUG [self d
我遇到错误,无法完成构建。我搜索了 Stackoverflow 和 Github。我已经尝试了很多方法,但我无法修复。请帮忙。 (1) 在 [src/nullnull/debug, src/debug
我刚刚意识到,使用 TFS 部署时,DEBUG 处理器指令仍然有效,有没有办法更改 TFS/Azure 网站或构建定义中的设置,而不是在本地解决方案配置? 我仍然希望本地解决方案保持调试状态,只有部署
我有一段代码在 VS2008,C++ 中以 Debug模式运行。 问题是,当我逐行调试代码时,在代码的一个非常奇怪的地方,它崩溃并说: debug assertion faild. Expressio
我有一个简单的 Xamarin.Forms 项目,我在 Visual Studio 中运行,使用 iphone 模拟器。我在 App.cs 中有以下代码: protected override voi
我是一名优秀的程序员,十分优秀!