How to catch a Swift crash and do some logging

2019-02-16 11:05发布

In Objective-C, the NSSetUncaughtExceptionHandler can register some functionality to do some last minute logging about the exception.

This doesn't catch something crashing from Swift.

Is it possible to do something like this at a global level in Swift? E.g. do some logging if a crash happens in Swift code, like forced unwrapping a nil optional.

Specifically, I'm doing a utility to log network traffic in the app, and I would like to flush my in-memory data to disk if a crash happens.

3条回答
Juvenile、少年°
2楼-- · 2019-02-16 11:30

NSSetUncaughtExceptionHandler works only with NSExceptions. See this SO answer for brilliant explanation.

To catch Swift run time errors implement signal handler for SIGTRAP. As far as I know, Swift code will terminate the program with SIGTRAP exception type if it detects an unexpected condition at runtime i.e only SIGTRAP is helpful to catch Swift errors, rest like SIGSEGV, SIGBUS, SIGILL do not work. I found this info in this apple link.

If your code is a mix both Objective-C and Swift, then implement both NSSetUncaughtExceptionHandler and signal handler to handle crashes.

For understanding and implementing Signal handling refer this link.

Hope this helps.

查看更多
神经病院院长
3楼-- · 2019-02-16 11:34

There are multiple aspects to your questions, let me try to answer them:

  1. NSSetUncaughtExceptionHandler only catches uncaught exceptions which is only a small subset of possible crashes.
  2. Exceptions in Objective-C are defined to be fatal and it is not recommended to catch them and even more not recommended to run any non-async safe code, which includes any Objective-C and Swift code
  3. To catch crashes you need to setup signal handlers and then only run async-safe code at the time of the crash, which is only a small subset of C. Especially you may not allocate new memory at crash time at all.
  4. Please remember, when your app crashed it is in a highly unstable code and object or variable references may point to something completely unexpected. So you really shouldn't do anything at crash time at all.
  5. The best open source way to work with crashes is using PLCrashReporter, which also provides method to invoke code at crash time. But again, this has to be async-safe.

To your specific scenario: you should not do that and not try to. This may cause user data to get corrupted or more data to be lost. The app crashed hard with serious issues, rather fix the issue.

查看更多
地球回转人心会变
4楼-- · 2019-02-16 11:37
NSSetUncaughtExceptionHandler(&HandleException);
signal(SIGABRT, SignalHandler);
signal(SIGILL, SignalHandler);
signal(SIGSEGV, SignalHandler);
signal(SIGFPE, SignalHandler);
signal(SIGBUS, SignalHandler);
signal(SIGPIPE, SignalHandler);

This will works for most situations. But I'm also trying to find some way to catch all crashes. Well, if you comd+click to those signals to see documents, you will find there are more than 20 kinds of signals, what I've done is signal(,) all most all kinds of signals. That seems works, but maybe still some crash cannot collect.

查看更多
登录 后发表回答