热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

iOS中程序异常Crash友好化处理详解

在iOS开发调试过程中以及上线之后,程序经常会出现崩溃的问题,下面这篇文章主要给大家介绍了关于iOS中程序异常Crash友好化处理的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考下

前言

前两天接到个面试,面试官问到上线的app怎么避免闪退,首先想到的就是在编码的时候进行各种容错,但貌似并不是面试官想要的答案,所以表现的很糟糕。今天有时间就来整理一下,希望有所帮助。

实现效果如图:


效果实现:


用法:

1.将截图的中CatchedHelper文件夹拖到你的项目工程中。

2.在AppDelegate.m中找到以下方法并如下添加代码:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
 // Override point for customization after application launch.
 
 [UncaughtExceptionHandler installUncaughtExceptionHandler:YES showAlert:YES];
 return YES;
}

以上代码就可以实现稍微友好一点的crash拦截处理。

代码解释:

UncaughtExceptionHandler.h主要代码:

#import 
#import 

@interface UncaughtExceptionHandler : NSObject

/*!
 * 异常的处理方法
 *
 * @param install 是否开启捕获异常
 * @param showAlert 是否在发生异常时弹出alertView
 */
+ (void)installUncaughtExceptionHandler:(BOOL)install showAlert:(BOOL)showAlert;
@end

UncaughtExceptionHandler.m文件主要的代码如下:

1.发送异常信号

/*
 * 异常的处理方法
 *
 * @param install  是否开启捕获异常
 * @param showAlert 是否在发生异常时弹出alertView
 */
+ (void)installUncaughtExceptionHandler:(BOOL)install showAlert:(BOOL)showAlert {
  
  if (install && showAlert) {
    [[self alloc] alertView:showAlert];
  }
  
  NSSetUncaughtExceptionHandler(install ? HandleException : NULL);
  signal(SIGABRT, install ? SignalHandler : SIG_DFL);
  signal(SIGILL, install ? SignalHandler : SIG_DFL);
  signal(SIGSEGV, install ? SignalHandler : SIG_DFL);
  signal(SIGFPE, install ? SignalHandler : SIG_DFL);
  signal(SIGBUS, install ? SignalHandler : SIG_DFL);
  signal(SIGPIPE, install ? SignalHandler : SIG_DFL);
}

产生上述的signal的时候就会调用我们定义的SignalHandler来处理异常。

ps: NSSetUncaughtExceptionHandler就是iOS SDK中提供的一个现成的函数,用来捕获异常的方法,使用方便。但它不能捕获抛出的signal,所以定义了SignalHandler方法。

2.处理异常

void HandleException(NSException *exception) {
  
  
  int32_t exceptiOnCount= OSAtomicIncrement32(&UncaughtExceptionCount);
  // 如果太多不用处理
  if (exceptionCount > UncaughtExceptionMaximum) {
    return;
  }
  
  //获取调用堆栈
  NSArray *callStack = [exception callStackSymbols];
  NSMutableDictionary *userInfo = [NSMutableDictionary dictionaryWithDictionary:[exception userInfo]];
  [userInfo setObject:callStack forKey:UncaughtExceptionHandlerAddressesKey];
  
  //在主线程中,执行制定的方法, withObject是执行方法传入的参数
  [[[UncaughtExceptionHandler alloc] init]
   performSelectorOnMainThread:@selector(handleException:)
   withObject:
   [NSException exceptionWithName:[exception name]
               reason:[exception reason]
              userInfo:userInfo]
   waitUntilDone:YES];
}

该方法就是对应NSSetUncaughtExceptionHandler的处理,只要方法关联到这个函数,那么发生相应错误时会自动调用该函数,调用时会传入exception参数。获取异常后会将捕获的异常传入最终调用处理的handleException函数。

3.无法捕获的signal处理

//处理signal报错
void SignalHandler(int signal) {
  
  int32_t exceptiOnCount= OSAtomicIncrement32(&UncaughtExceptionCount);
  // 如果太多不用处理
  if (exceptionCount > UncaughtExceptionMaximum) {
    return;
  }
  
  NSString* description = nil;
  switch (signal) {
    case SIGABRT:
      description = [NSString stringWithFormat:@"Signal SIGABRT was raised!\n"];
      break;
    case SIGILL:
      description = [NSString stringWithFormat:@"Signal SIGILL was raised!\n"];
      break;
    case SIGSEGV:
      description = [NSString stringWithFormat:@"Signal SIGSEGV was raised!\n"];
      break;
    case SIGFPE:
      description = [NSString stringWithFormat:@"Signal SIGFPE was raised!\n"];
      break;
    case SIGBUS:
      description = [NSString stringWithFormat:@"Signal SIGBUS was raised!\n"];
      break;
    case SIGPIPE:
      description = [NSString stringWithFormat:@"Signal SIGPIPE was raised!\n"];
      break;
    default:
      description = [NSString stringWithFormat:@"Signal %d was raised!",signal];
  }
  
  NSMutableDictionary *userInfo = [NSMutableDictionary dictionary];
  NSArray *callStack = [UncaughtExceptionHandler backtrace];
  [userInfo setObject:callStack forKey:UncaughtExceptionHandlerAddressesKey];
  [userInfo setObject:[NSNumber numberWithInt:signal] forKey:UncaughtExceptionHandlerSignalKey];
  
  //在主线程中,执行指定的方法, withObject是执行方法传入的参数
  [[[UncaughtExceptionHandler alloc] init]
   performSelectorOnMainThread:@selector(handleException:)
   withObject:
   [NSException exceptionWithName:UncaughtExceptionHandlerSignalExceptionName
               reason: description
              userInfo: userInfo]
   waitUntilDone:YES];
}

以上方法是对于捕获不到的signal信号进行处理,列出常见的异常类型。

4.堆栈调用

//获取调用堆栈
+ (NSArray *)backtrace {
  
  //指针列表
  void* callstack[128];
  //backtrace用来获取当前线程的调用堆栈,获取的信息存放在这里的callstack中
  //128用来指定当前的buffer中可以保存多少个void*元素
  //返回值是实际获取的指针个数
  int frames = backtrace(callstack, 128);
  //backtrace_symbols将从backtrace函数获取的信息转化为一个字符串数组
  //返回一个指向字符串数组的指针
  //每个字符串包含了一个相对于callstack中对应元素的可打印信息,包括函数名、偏移地址、实际返回地址
  char **strs = backtrace_symbols(callstack, frames);
  
  int i;
  NSMutableArray *backtrace = [NSMutableArray arrayWithCapacity:frames];
  for (i = 0; i 

backtrace是Linux下用来追踪函数调用堆栈以及定位段错误的函数。

5.使用UIAlerView进行友好化提示

- (void)handleException:(NSException *)exception {
  
  [self validateAndSaveCriticalApplicationData:exception];
  
  if (!showAlertView) {
    return;
  }
  
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
  UIAlertView *alert =
  [[UIAlertView alloc]
   initWithTitle:@"出错啦"
   message:[NSString stringWithFormat:@"你可以尝试继续操作,但是应用可能无法正常运行.\n"]
   delegate:self
   cancelButtonTitle:@"退出"
   otherButtonTitles:@"继续", nil];
  [alert show];
#pragma clang diagnostic pop
  
  CFRunLoopRef runLoop = CFRunLoopGetCurrent();
  CFArrayRef allModes = CFRunLoopCopyAllModes(runLoop);
  
  while (!self.dismissed) {
    //点击继续
    for (NSString *mode in (__bridge NSArray *)allModes) {
      //快速切换Mode
      CFRunLoopRunInMode((CFStringRef)mode, 0.001, false);
    }
  }
  
  //点击退出
  CFRelease(allModes);
  
  NSSetUncaughtExceptionHandler(NULL);
  signal(SIGABRT, SIG_DFL);
  signal(SIGILL, SIG_DFL);
  signal(SIGSEGV, SIG_DFL);
  signal(SIGFPE, SIG_DFL);
  signal(SIGBUS, SIG_DFL);
  signal(SIGPIPE, SIG_DFL);
  
  if ([[exception name] isEqual:UncaughtExceptionHandlerSignalExceptionName]) {
    
    kill(getpid(), [[[exception userInfo] objectForKey:UncaughtExceptionHandlerSignalKey] intValue]);
    
  } else {
    
    [exception raise];
  }
}

在这里你可以做自己的crash收集操作,例如上传服务器等。

源码下载

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。


推荐阅读
  • 本文详细介绍了JSP(Java Server Pages)的九大内置对象及其功能,探讨了JSP与Servlet之间的关系及差异,并提供了实际编码示例。此外,还讨论了网页开发中常见的编码转换问题以及JSP的两种页面跳转方式。 ... [详细]
  • NFS(Network File System)即网络文件系统,是一种分布式文件系统协议,主要用于Unix和类Unix系统之间的文件共享。本文详细介绍NFS的配置文件/etc/exports和相关服务配置,帮助读者理解如何在Linux环境中配置NFS客户端。 ... [详细]
  • 本文探讨了Web开发与游戏开发之间的主要区别,旨在帮助开发者更好地理解两种开发领域的特性和需求。文章基于作者的实际经验和网络资料整理而成。 ... [详细]
  • 在Linux系统上构建Web服务器的详细步骤
    本文详细介绍了如何在Linux系统上搭建Web服务器的过程,包括安装Apache、PHP和MySQL等关键组件,以及遇到的一些常见问题及其解决方案。 ... [详细]
  • 本文将详细介绍如何在ThinkPHP6框架中实现多数据库的部署,包括读写分离的策略,以及如何通过负载均衡和MySQL同步技术优化数据库性能。 ... [详细]
  • JavaWeb技术架构解析
    本文探讨了JavaWeb开发中客户端与服务器端的交互模式,重点分析了B/S(浏览器/服务器)和C/S(客户端/服务器)两种架构的特点及应用场景。 ... [详细]
  • iTOP4412开发板QtE5.7源码编译指南
    本文详细介绍了如何在iTOP4412开发板上编译QtE5.7源码,包括所需文件的位置、编译器设置、触摸库编译以及QtE5.7的完整编译流程。 ... [详细]
  • Eclipse 中 JSP 开发环境配置指南
    本文详细介绍了如何在 Eclipse 集成开发环境中配置 JSP 运行环境,包括必要的软件下载、Tomcat 服务器的配置以及常见问题的解决方法。 ... [详细]
  • HTTPS与TLS/SSL协议详解:握手及记录协议
    HTTPS,即HTTP over TLS/SSL,通过在HTTP通信层引入安全协议,确保数据传输的安全性。本文将深入探讨TLS/SSL协议的基本概念、HTTPS的必要性,以及TLS握手和记录协议的工作原理。 ... [详细]
  • 应对.avast后缀勒索病毒:全面指南
    本文详细介绍了.avast后缀勒索病毒的特性、感染途径、恢复方法及预防措施,旨在帮助用户有效应对这一威胁。 ... [详细]
  • 解决CentOS 7.5中无法通过man命令查询C语言库函数的问题
    本文描述了在CentOS 7.5操作系统上,使用man命令查询C语言库函数时遇到的问题,并提供了详细的解决方案。 ... [详细]
  • 在安装Ubuntu 12.10并尝试安装VMware Tools时,遇到了一个常见的错误提示:指定的路径不是有效的3.5.0-17-generic内核头文件路径。本文将提供解决这一问题的具体步骤。 ... [详细]
  • 如何在Linux中实现字符设备控制
    本文详细探讨了在Linux环境下控制字符设备的方法,包括蜂鸣器和模数转换器(ADC)的实际操作案例。对于开发者来说,了解这些基础知识对于嵌入式系统的开发尤为重要。 ... [详细]
  • Linux中为何使用chmod 777设置最大文件权限
    在Linux系统管理中,设置文件权限是一个常见的操作。其中,chmod 777命令用于赋予文件所有者、组和其他用户完全控制权限。本文将探讨这一命令背后的原理及其应用。 ... [详细]
  • 本文详细探讨了Android系统中的内部存储路径,包括如何正确地使用这些路径进行文件操作,以及不同存储路径的特点和权限要求。 ... [详细]
author-avatar
fghnh102_441
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有