欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2024-02-08 10:45:40
前言 在平时自己调试的时候,可以直接连接电脑,直接在窗口中查看结果。但是在测试人员测试,或者灰度测试的时候,怎么才能拿到日志呢?最先想到的肯定是输出到本地文件,然后在需要...

前言

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

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

     – 把日志输出到文件中

     – 捕捉异常信息

【解析都写在注释中了】

示例代码

- (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开发者们能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持。