-(void)saveDEBUGlog{ NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES); NSString *documentDirectory = [paths objectAtIndex:0]; NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init]; [dateFormatter setDateFormat:@"yyyy_MM_dd_HH_mm_ss"]; NSString *currentDateStr = [dateFormatter stringFromDate:[NSDate date]]; NSString *fileName = [NSString stringWithFormat:@"testLog_%@.log",currentDateStr]; NSString *logFilePath = [documentDirectory stringByAppendingPathComponent:fileName]; // 先刪除已經存在的文件 NSFileManager *defaultManager = [NSFileManager defaultManager]; [defaultManager removeItemAtPath:logFilePath error:nil]; // 將log輸入到文件 freopen([logFilePath cStringUsingEncoding:NSASCIIStringEncoding], "a+", stdout); freopen([logFilePath cStringUsingEncoding:NSASCIIStringEncoding], "a+", stderr);}
這個方法主要是調用 freopen 這個方法來寫入, 其中 stdout 和 stderr 囊括了 iOS 大部分的異常輸出。
2. 根據 Bool 值控制 log 輸出
用戶在使用 app 遇到各種各樣的問題,當自己以及測試團隊不好定位原因的時候,能將用戶把關鍵點的 log 發送過來是最好的分析方法了。但是如何將 app 運行過程中的 log 截取一部分保存呢?像開關一樣能夠控制 log 的讀寫呢?通過閱讀 MQTTLog 源碼發現獲得的靈感。
首先對 NSLog 進行下宏替換,項目中統一使用 SLOG 來進行輸出
#define SLOG(fmt, ...) if (reportLoggerIsOpen) NSLog((@"%s [Line %d] " fmt), __PRETTY_FUNCTION__, __LINE__, ##__VA_ARGS__)
從宏定義可以看出,reportLoggerIsOpen 是控制 Log 輸出的開關,對于reportLoggerIsOpen的定義,在 h 文件中先 extern 聲明這個變量,然后再 m 文件中去實現。
.h 文件
#import <Foundation/Foundation.h>extern BOOL reportLoggerIsOpen;.m 文件BOOL reportLoggerIsOpen = NO;+ (void)setLogOpen:(BOOL)open { reportLoggerIsOpen = open;}
通過 setLogOpen 這個方法,就能夠收放自如的控制日志寫入了。比如你需要抓取登錄模塊的日志,那么就在登錄前傳入 true,登錄完畢后,傳入 false,即可只保留登錄模塊的日志了。
新聞熱點
疑難解答