回溯会错过导致崩溃的函数



当我的应用程序崩溃时,我使用回溯backtrace_symbols来收集调用堆栈。

void InstallSignalHandler(void)
{
signal(SIGHUP, SignalExceptionHandler);
signal(SIGINT, SignalExceptionHandler);
signal(SIGQUIT, SignalExceptionHandler);
signal(SIGABRT, SignalExceptionHandler);
signal(SIGILL, SignalExceptionHandler);
signal(SIGSEGV, SignalExceptionHandler);
signal(SIGFPE, SignalExceptionHandler);
signal(SIGBUS, SignalExceptionHandler);
signal(SIGPIPE, SignalExceptionHandler);
}
void SignalExceptionHandler(int signal)
{
NSMutableString *mstr = [[NSMutableString alloc] init];
[mstr appendString:@"Stack:n"];
void* callstack[128];
int i, frames = backtrace(callstack, 128);
char** strs = backtrace_symbols(callstack, frames);
for (i = 0; i <frames; ++i) {
[mstr appendFormat:@"%sn", strs[i]];
}
NSLog(@"%@", mstr);
free(strs);
}

当我检查登录控制台时,我只是发现日志错过了导致崩溃的功能。函数为:

+ (void)testCrash
{
int *nullPointer = NULL;
*nullPointer = 2019;
}

登录控制台是:

0   TestApp                             0x0000000101d1e040 SignalExceptionHandler + 160
1   libsystem_platform.dylib            0x000000011002bb5d _sigtramp + 29
2   ???                                 0x0000000000000000 0x0 + 0
3   TestApp                             0x00000001019bbc6f __39+[MyCrashTesting showInViewController:]_block_invoke + 303
4   UIKit                               0x000000010b09a559 -[UIAlertController _invokeHandlersForAction:] + 105
5   UIKit                               0x000000010b09af5e __103-[UIAlertController _dismissAnimated:triggeringAction:triggeredByPopoverDimmingView:dismissCompletion:]_block_invoke.461 + 16
6   UIKit                               0x000000010ae42ca2 -[UIPresentationController transitionDidFinish:] + 1346
7   UIKit                               0x000000010ae46b12 __56-[UIPresentationController runTransitionForCurrentState]_block_invoke.436 + 183
8   UIKit                               0x000000010ba2a3b4 -[_UIViewControllerTransitionContext c

我认为函数名称"testCrash"应该在日志的顶部。我做错了什么吗?

由于这是一个非常简单的方法,没有参数,也没有ARC在clang内部繁重的工作,因此Objective-C编译器能够进行优化(-O1足以实现这一点(删除消息运行时调用

您可以通过以下方式防止此行为:

+ (void)testCrash __attribute__((optnone))
{
int *nullPointer = NULL;
*nullPointer = 2019;
}

这里学到的教训是,在程序执行期间,您永远不应该依赖特定的堆栈跟踪来实现某些目标。只要您了解刚刚遇到的优化原则,诊断就可以了。

最新更新