iOS 符号化方法

介绍 iOS 堆栈符号化常用的方法。

符号化可以将上面的得到的代码内存地址转为可读的函数签名信息,包括函数名、文件名、行数等等,方便我们快速定位问题。

符号化常用方法

使用 backtrace_symbols_fd 方法进行运行时符号化

backtrace_symbols_fd 函数能够将从backtrace 中得到的原始堆栈信息进行符号化,并输出到控制台。

vm_address_t *stacks_2[128];
int depth = backtrace((void **)stacks_2, 128);
backtrace_symbols_fd((void **)stacks_2, depth, STDOUT_FILENO);     // STDOUT_FILENO 表示向屏幕输出
💡
注意,这里用的是 backtrace_symbols_fd 而不是 backtrace_symbols。backtrace_symbols 内部实现也有 alloc/malloc 相关的调用,在 malloc_logger 里面调用会陷入递归死循环

使用 atos 和 dSYM 对堆栈信息符号化

假设我们的崩溃日志系统收集到了下面这样的崩溃日志:

"0 libsystem_kernel.dylib 0x1b00df000 0x1b00e1d0c" "1 libsystem_kernel.dylib 0x1b00df000 0x1b00ffe70" "2 Keep 0x104584000 0x10704e864" "3 Keep 0x104584000 0x10704ac5c" "4 libsystem_malloc.dylib 0x192e77000 0x192e89b4c" "5 Keep 0x104584000 0x10a1d9c2c" "6 Keep 0x104584000 0x10a209ba4" "7 Keep 0x104584000 0x10a209e48" "8 Keep 0x104584000 0x108c877b8" "9 Keep 0x104584000 0x108c86c04" "10 Keep 0x104584000 0x108c6c8d0" "11 Keep 0x104584000 0x108c68938" "12 Keep 0x104584000 0x108c664d8" "13 Keep 0x104584000 0x108c66d5c" "14 Keep 0x104584000 0x108c66ee8" "15 libdispatch.dylib 0x183f33000 0x183f932b0" "16 libdispatch.dylib 0x183f33000 0x183f94298" "17 libdispatch.dylib 0x183f33000 0x183f3ca40" "18 libdispatch.dylib 0x183f33000 0x183f3d518" "19 libdispatch.dylib 0x183f33000 0x183f46fac" "20 libsystem_pthread.dylib 0x1cc79e000 0x1cc7a95bc" 

我们需要先拿到 App 的 dSYM 文件。之后在终端使用 xcrun atos 进行符号化。

xcrun atos -arch arm64 -o Keep.app.dSYM/Contents/Resources/DWARF/Keep -l 0x104584000
0x108c66ee8
# 第一行指令输入一次即可,随后可以逐次输入调用方法的内存地址

逐次输入各个调用方法的内存地址后,控制台会输出相应的解析结果。

符号化

动态库堆栈符号化

iOS 静态库是在程序编译的时候就进行了链接,拷贝到了 App 的可执行文件里。而动态库并不会在编译时置入 App 中,而是在 App 运行的时候去动态查找动态库并进行链接。

静态库代码发生崩溃时,堆栈信息也是和 App 放在一起的,内存加载地址一致,可以用 dSYM 解析。

但是动态库堆栈信息无法使用 dSYM 解析,需要利用动态库的 framework 进行解析。

堆栈信息:

"0 libdispatch.dylib 0x1210e4000 0x1210eabb8" "5 KPlayerCore 0x121f48000 0x121f61134" "6 KPlayerCore 0x121f48000 0x121f954a8" "7 KPlayerCore 0x121f48000 0x121f95634" "8 KPlayerCore 0x121f48000 0x121f74e5c" "9 libsystem_pthread.dylib 0x213dcc000 0x213dcd6b8"

找到工程使用的动态库 framework:

img.png

使用 atos 工具解析堆栈:

img.png

Read more

2025 关税危机中学到的投资经验

2025 关税危机中学到的投资经验

充足的现金流很重要 好的买入机会不会每天都出现,但当它出现的时候,你最好还有筹码可以投入。 有些人手里握不住钱,一有闲钱就赶紧买入基金、股票,生怕错过了机会,让钱白搭手里。市场是疯狂的、充满变数的,尤其是在特朗普上台后,一句话就可能让股市涨停或跌停。那些专业的理财投资机构尚不能预测市场,何况我们这些散户呢。在不稳定的市场中,我们要学习巴菲特,备好现金,耐心等待买入(抄底)机会。 不要提前打光子弹 美股标普 500 指数从 2 月中旬到 3 月中旬累计跌了约 10%。如果这时候你觉得已经跌了很多,可以 all in 抄底了,那么你就会错过 4 月上旬的那次狂跌——一周跌了约 10%。没有人能预测市场,除了此刻的股市指挥家特朗普。散户们能学到的经验就是「永远不要提前打光子弹」,你以为的谷底其实只是个半山腰。 相信自己,保持耐心 在美股大跌的时段里,小红书、v2ex

By Gray
SwiftUI 页面导航最佳实践

SwiftUI 页面导航最佳实践

通过全局 Router 1. 定义一个全局 Router 对象,维护页面跳转类型和参数。 @Observable final class Router { public enum Destination: Codable, Hashable { case pageA(models: [Model]) case pageB } var navPath = NavigationPath() func navigate(to destination: Destination) { navPath.append(destination) } func navigateBack() { navPath.removeLast() } func navigateToRoot() { navPath.removeLast(navPath.count) } } 枚举 Destination 可以指

By Gray
碎碎念——投资,不确定性沟通定语

碎碎念——投资,不确定性沟通定语

投资理财 最近因为关税的冲击,美股正在经历一波大跌行情。我个人比较看好纳斯达克,也在一直定投纳斯达克。我是长期主义者,没有精力和时间在短期波动中挣钱,只想在下跌调整中「进货」。 定投分左侧定投和右侧定投。左侧定投是在下跌的过程中定投,而右侧定投是在上涨的过程中定投。左侧定投无法确认底部在哪里,需要源源不断往里投入金钱(行内成为「子弹」);右侧定投无法确认反弹是诱多还是形势已经逆转。我采用的是左侧定投,大跌大加,小跌小加,反弹时停止定投。不论采用哪种定投,殊途同归,都是尽量降低投资成本。 目前网上看衰美股的声音不少,不少人因为恐慌割肉卖出股票。但我们要知道目前美国仍旧是世界第一大国,消费潜力巨大,大型科技公司(苹果、英伟达等)的基本面并没有出现大问题。只是因为特朗普的「量子态」关税政策,导致市场恐慌抛售。我们无需担心纳斯达克、标普指数从此一蹶不振。恰恰相反,现在是买入美股的绝佳时机。苹果、英伟达等大型公司的 PE 值已经降到了合理位置,只要不买妖股,不投机,只关注纳斯达克、标普指数,只买大型公司股票,迟早会取得丰厚盈利的。

By Gray
怀念小时候吃过的食物

怀念小时候吃过的食物

前两天下班骑车回家的路上听到了路旁有人在讨论泡馍。他们口中的泡馍应该是类似西安羊肉泡馍之类的食物。但是我却想起来了小时候吃的不一样的泡馍以及其他吃食。 不一样的泡馍 小时候我们那里普遍比较贫穷,家家户户除了过年过节基本上很难吃到大块肉。小孩子饭量时小时大,中午吃的饭,半晌就又饿了。家里有大葱或者豆糁的话,可以拿着一个馍就着就吃了。整根的葱是最下馍的,葱白部分甜又辣,葱叶里面会有像鼻涕一样的粘液,要把它挤出来才下得嘴吃。豆糁是黄豆的发酵产物,煮熟的大豆加盐发酵几天,黏丝丝的时候团成球,放到发黑就能吃了。吃的时候从球上掰下来几小块就行。豆糁是咸的,因而也能下饭。不过最妙的吃法是将豆糁和鸡蛋一起炒。鸡蛋的香气和豆糁稍微发臭的味道混在一起,形成一种独特的香味。像北京的臭豆腐一样,闻着臭,吃着香。 如果家里没葱没豆糁了,馍又很干,那泡馍就是解决饿肚子的绝好办法。将干硬的馍掰成几瓣,不能太碎小,放到瓷碗里。倒入炒菜的肉味王佐料,或者是平时攒下来的方便面调料。再提溜着暖水瓶,倒进去冒着热气的水。当然香油是少不了的,拿着油光光的瓶子,滴进去几滴喷香的香油。最后用大碗盖住,或者干脆啥也不盖,静等

By Gray