全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

iOS中捕获日志与异常示例详解

前言

在平时自己调试的时候,可以直接连接电脑,直接在窗口中查看结果。但是在测试人员测试,或者灰度测试的时候,怎么才能拿到日志呢?最先想到的肯定是输出到本地文件,然后在需要的时候进行上传。

分享一段之前找到的方法,下面的代码提供了两个主要功能:

     – 把日志输出到文件中

     – 捕捉异常信息

【解析都写在注释中了】

示例代码

- (void)redirectNSLogToDocumentFolder
{
//如果已经连接Xcode调试则不输出到文件
//该函数用于检测输出 (STDOUT_FILENO) 是否重定向 是个 Linux 程序方法
if(isatty(STDOUT_FILENO)) {
return;
}

// 判断 当前是否在 模拟器环境 下 在模拟器不保存到文件中
UIDevice *device = [UIDevice currentDevice];
if([[device model] hasSuffix:@"Simulator"]){
return;
}

//将NSlog打印信息保存到Document目录下的Log文件夹下
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *logDirectory = [[paths objectAtIndex:0] stringByAppendingPathComponent:@"Log"];

NSFileManager *fileManager = [NSFileManager defaultManager];
BOOL fileExists = [fileManager fileExistsAtPath:logDirectory];
if (!fileExists) {
[fileManager createDirectoryAtPath:logDirectory withIntermediateDirectories:YES attributes:nil error:nil];
}

NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
[formatter setLocale:[[NSLocale alloc] initWithLocaleIdentifier:@"zh_CN"]];
[formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"]; //每次启动后都保存一个新的日志文件中
NSString *dateStr = [formatter stringFromDate:[NSDate date]];
NSString *logFilePath = [logDirectory stringByAppendingFormat:@"/%@.log",dateStr];

// 将log输入到文件
freopen([logFilePath cStringUsingEncoding:NSUTF8StringEncoding], "a+", stdout);
freopen([logFilePath cStringUsingEncoding:NSUTF8StringEncoding], "a+", stderr);

//未捕获的Objective-C异常日志
NSSetUncaughtExceptionHandler (&UncaughtExceptionHandler);
}

之前看的时候,对 NSSetUncaughtExceptionHandler(&UncaughtExceptionHandler) 这个用法一知半解,去翻了一下源码,这个方法是在 Foundation 中。

api 中的定义是Changes the top-level error handler ,Sets the top-level error-handling function where you can perform last-minute logging before the program terminates. 通过替换掉最高级别的 handle 方法,可以在程序终止之前可以获取到崩溃信息,并执行相应的操作,比如保存本地,或者上报。

方法调用为:

void NSSetUncaughtExceptionHandler(NSUncaughtExceptionHandler *);

传入的是一个 NSUncaughtExceptionHandler 的指针。

typedef void NSUncaughtExceptionHandler(NSException *exception);

意思就是需要一个 返回 void 并且参数为 NSException *exception 的函数指针。

你想要,那我就给你!

所以下面有个 C 语言的函数,你看这个写法和 OC 的声明也不一样。

void UncaughtExceptionHandler(NSException* exception)
{
NSString* name = [ exception name ];
NSString* reason = [ exception reason ];
NSArray* symbols = [ exception callStackSymbols ]; // 异常发生时的调用栈
NSMutableString* strSymbols = [ [ NSMutableString alloc ] init ]; //将调用栈拼成输出日志的字符串
for ( NSString* item in symbols )
{
[ strSymbols appendString: item ];
[ strSymbols appendString: @"\r\n" ];
}

//将crash日志保存到Document目录下的Log文件夹下
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *logDirectory = [[paths objectAtIndex:0] stringByAppendingPathComponent:@"Log"];

NSFileManager *fileManager = [NSFileManager defaultManager];
if (![fileManager fileExistsAtPath:logDirectory]) {
[fileManager createDirectoryAtPath:logDirectory withIntermediateDirectories:YES attributes:nil error:nil];
}

NSString *logFilePath = [logDirectory stringByAppendingPathComponent:@"UncaughtException.log"];
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
[formatter setLocale:[[NSLocale alloc] initWithLocaleIdentifier:@"zh_CN"]];
[formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
NSString *dateStr = [formatter stringFromDate:[NSDate date]];

NSString *crashString = [NSString stringWithFormat:@"<- %@ ->[ Uncaught Exception ]\r\nName: %@, Reason: %@\r\n[ Fe Symbols Start ]\r\n%@[ Fe Symbols End ]\r\n\r\n", dateStr, name, reason, strSymbols];
//把错误日志写到文件中
if (![fileManager fileExistsAtPath:logFilePath]) {
[crashString writeToFile:logFilePath atomically:YES encoding:NSUTF8StringEncoding error:nil];
}else{
NSFileHandle *outFile = [NSFileHandle fileHandleForWritingAtPath:logFilePath];
[outFile seekToEndOfFile];
[outFile writeData:[crashString dataUsingEncoding:NSUTF8StringEncoding]];
[outFile closeFile];
}

//把错误日志发送到邮箱
// NSString *urlStr = [NSString stringWithFormat:@"mailto://XXXXX@126.com?subject=bug报告&body=感谢您的配合!<br><br><br>错误详情:<br>%@",crashString ];
// NSURL *url = [NSURL URLWithString:[urlStr stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]];
// [[UIApplication sharedApplication] openURL:url];
}

总结

以上就是这篇文章的全部内容了,希望本文的内容对各位iOS开发者们能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。


# ios  # 异常捕获  # 日志分析  # 崩溃日志捕获  # iOS横屏弹键盘的高度错误异常解决  # iOS中程序异常Crash友好化处理详解  # iOS下拉刷新 UIScrollVie异常闪动问题  # 解决iOS11图片下拉放大出现信号栏白条的bug问题  # iOS之异常与信号使用场景分析  # 的是  # 您的  # 是个  # 也不  # 是在  # 给你  # 有个  # 目录下  # 你看  # 可以直接  # 中了  # 写到  # 发送到  # 这篇文章  # 翻了  # 谢谢大家  # 写在  # 主要功能  # 你想要  # 不保存 


相关文章: 深圳防火门网站制作公司,深圳中天明防火门怎么编码?  网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?  电视网站制作tvbox接口,云海电视怎样自定义添加电视源?  实现虚拟支付需哪些建站技术支撑?  成都响应式网站开发,dw怎么把手机适应页面变成网页?  山东网站制作公司有哪些,山东大源集团官网?  网站制作怎么样才能赚钱,用自己的电脑做服务器架设网站有什么利弊,能赚钱吗?  电脑免费海报制作网站推荐,招聘海报哪个网站多?  西安大型网站制作公司,西安招聘网站最好的是哪个?  红河网站制作公司,红河事业单位身份证如何上传?  浅析上传头像示例及其注意事项  php能控制zigbee模块吗_php通过串口与cc2530 zigbee通信【介绍】  C#如何序列化对象为XML XmlSerializer用法  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  电影网站制作价格表,那些提供免费电影的网站,他们是怎么盈利的?  c# 在ASP.NET Core中管理和取消后台任务  厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?  南平网站制作公司,2025年南平市事业单位报名时间?  如何快速登录WAP自助建站平台?  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?  建站三合一如何选?哪家性价比更高?  香港服务器租用每月最低只需15元?  如何配置FTP站点权限与安全设置?  测试制作网站有哪些,测试性取向的权威测试或者网站?  建站之星备案流程有哪些注意事项?  如何解决ASP生成WAP建站中文乱码问题?  建站之星如何配置系统实现高效建站?  建站一年半SEO优化实战指南:核心词挖掘与长尾流量提升策略  建站主机与服务器功能差异如何区分?  网站建设制作、微信公众号,公明人民医院怎么在网上预约?  如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法  临沂网站制作公司有哪些,临沂第四中学官网?  Dapper的Execute方法的返回值是什么意思 Dapper Execute返回值详解  设计网站制作公司有哪些,制作网页教程?  建站主机如何选?性能与价格怎样平衡?  如何挑选最适合建站的高性能VPS主机?    建站之星会员如何解锁更多建站功能?  小程序网站制作需要准备什么资料,如何制作小程序?  建站之星如何快速解决建站难题?  如何使用Golang table-driven基准测试_多组数据测量函数效率  如何选择美橙互联多站合一建站方案?  如何通过二级域名建站提升品牌影响力?  免费ppt制作网站,有没有值得推荐的免费PPT网站?  建站10G流量真的够用吗?如何应对访问高峰?  如何在橙子建站上传落地页?操作指南详解  道歉网站制作流程,世纪佳缘致歉小吴事件,相亲网站身份信息伪造该如何稽查?  已有域名如何快速搭建专属网站?  移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?  如何快速查询网站的真实建站时间? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。