iOS - 原理探究之Log的重定向

Created by Shuimu

日志分級

分級定義:

Fatal 嚴(yán)重級別讲逛。

表明程序遇到了非常嚴(yán)重的錯誤悉尾,已經(jīng)不能繼續(xù)運行下去檐晕。

Error 出錯級別。

程序可能正處于非正常狀態(tài)下县恕,無法正常完成既定功能,但程序依然能運行剂桥。

Warning 警告級別忠烛。

程序處理中遇到非法數(shù)據(jù)或者某種可能的錯誤,程序無法處理权逗,但是依然能正常運行美尸。

Info 重要信息級別。

一般用于打印模塊中一些重要的點斟薇,以證明程序運行正常师坎。

Debug 調(diào)試級別。

表明當(dāng)前正在臨時打印一些log堪滨,以便調(diào)試程序胯陋。

輸出格式

[level] file function [line num] message

[level] file function [line num] message

[level] file function [line num] message

基本使用方法

像極了 NSLog

#define XXLogFatal(fmt, ...)   // 省略具體實現(xiàn)
#define XXLogError(fmt, ...)   // 省略具體實現(xiàn)
#define XXLogWarn(fmt, ...)    // 省略具體實現(xiàn)
#define XXLogInfo(fmt, ...)    // 省略具體實現(xiàn)
#define XXLogDebug(fmt, ...)   // 省略具體實現(xiàn)

#define XXLog XXLogDebug       // 等價于 Debug 級別(只是為了少打幾個字母)

日志讀取

簡介

日常開發(fā)中,經(jīng)常會用到NSLog進(jìn)行調(diào)試,NSLog本質(zhì)上是一個C函數(shù)惶岭,它的函數(shù)聲明如下:

FOUNDATION_EXPORT void NSLog(NSString *format, ...) NS_FORMAT_FUNCTION(1,2) NS_NO_TAIL_CALL;

系統(tǒng)的說明是:Logs an error message to the Apple System Log facility.寿弱。

它是用來輸出信息到標(biāo)準(zhǔn)Error控制臺的,內(nèi)部使用的是Apple System Log的API(相關(guān)API可以查看<asl.h>)按灶。調(diào)試階段症革,日志會輸出到Xcode控制臺中,真機調(diào)試時鸯旁,則會輸出到系統(tǒng)的/var/log/syslog這個文件中噪矛。

日志輸出到文件中的句柄定義在unistd.h文件中

#define  STDIN_FILENO   0   /* standard input file descriptor */
#define STDOUT_FILENO   1   /* standard output file descriptor */
#define STDERR_FILENO   2   /* standard error file descriptor */

NSLog的日志是通過STDERR_FILENO輸出到文件的。另外铺罢,fprintf并不會調(diào)用ASL接口艇挨,只是簡單的輸出信息(相對NSLog,少了日期韭赘、進(jìn)程名缩滨、進(jìn)程id等信息),其日志是通過STDOUT_FILENO輸出到文件中泉瞻。

ASL讀取日志

鑒于以上信息脉漏,既然日志是寫入到系統(tǒng)的syslog中,那么就可以直接讀取這些日志袖牙。通過ASL的相關(guān)API侧巨,我們可以讀取到這些日志內(nèi)容。

不過鞭达,查看<asl.h>相關(guān)API可以發(fā)現(xiàn)司忱,

__API_DEPRECATED("os_log(3) has replaced asl(3)", macosx(10.4,10.12), ios(2.0,10.0), watchos(2.0,3.0), tvos(9.0,10.0))

asl從iOS 10開始被棄用了,并提供了新的APIos_log進(jìn)行替代的(詳見<os/log.h>

??畴蹭,系統(tǒng)版本限制坦仍,放棄。

NSLog重定向

通過前面的介紹可以了解到叨襟,NSLog輸出的日志內(nèi)容繁扎,最終都通過STDERR句柄來記錄的,所以可以考慮對其進(jìn)行重定向(當(dāng)然芹啥,重定向之后锻离,就無法在控制臺看到log打印了)。

文件重定向

利用c語言的freopen函數(shù)進(jìn)行重定向墓怀,將寫往stderr的內(nèi)容重定向到我們指定的文件中去汽纠。

函數(shù)聲明:

FILE    *freopen(const char * __restrict, const char * __restrict, FILE * __restrict)
  • 第一個參數(shù):需要重定向到文件路徑
  • 第二個參數(shù):文件的打開模式,r/w/a/+等
  • 第三個參數(shù):被打開的文件名傀履,通常使用標(biāo)準(zhǔn)流文件(stdin/stdout/stderr)

核心代碼:

    NSString *newLogFilePath = @"xxxxxx"; // 指定的寫入文件路徑
    freopen([newLogFilePath cStringUsingEncoding:NSASCIIStringEncoding], "a+", stderr); // stderr 標(biāo)準(zhǔn)錯誤輸出流

由于iOS的沙盒機制虱朵,無法得知stderr原來的文件路徑莉炉,導(dǎo)致freopen無法重定向回去。這里利用dupdup2 來實現(xiàn)碴犬。

大致實現(xiàn)思路如下:

// 記錄原來的輸出流
int origin = dup(STDERR_FILENO);

// 利用freopen進(jìn)行重定向
FILE *myFile = freopen([newLogFilePath cStringUsingEncoding:NSASCIIStringEncoding], "a+", stderr);

// 恢復(fù)重定向
dup2(origin, STDERR_FILENO)

dup2重定向

上面提到的dup2函數(shù)是專門進(jìn)行文件重定向的絮宁。同樣也可以直接通過dup2函數(shù)直接重定向STDERR句柄,將內(nèi)容指向指定的位置服协。

  • 思路:

首先通過NSPipe創(chuàng)建一個管道绍昂,pipe有讀端和寫端,通過dup2將標(biāo)準(zhǔn)輸入重定向到pipe的寫端偿荷,在通過NSFileHandle監(jiān)聽pipe的讀端窘游,再處理讀出的信息。

  • 核心代碼如下:
- (void)redirectStandardOutput {
    // 記錄標(biāo)準(zhǔn)輸出及錯誤流原始文件描述符
    self.errFd = dup(STDERR_FILENO);
    stderr->_flags = 10;
    NSPipe *errPipe = [NSPipe pipe];
    NSFileHandle *pipeErrHandle = [errPipe fileHandleForReading];
    dup2([[errPipe fileHandleForWriting] fileDescriptor], STDERR_FILENO);
    [pipeErrHandle readInBackgroundAndNotify];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(redirectErrNotificationHandle:) name:NSFileHandleReadCompletionNotification object:pipeErrHandle];
    
}

- (void)recoverStandardOutput {
    dup2(self.errFd, STDERR_FILENO);
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

// 重定向之后的錯誤輸出
- (void)redirectErrNotificationHandle:(NSNotification *)nf {
    NSData *data = [[nf userInfo] objectForKey:NSFileHandleNotificationDataItem];
    NSString *str = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
    
    // write your code. 保存日志跳纳、上傳或展示等等
    
    
    [[nf object] readInBackgroundAndNotify];
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末忍饰,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子寺庄,更是在濱河造成了極大的恐慌艾蓝,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,907評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件斗塘,死亡現(xiàn)場離奇詭異赢织,居然都是意外死亡,警方通過查閱死者的電腦和手機逛拱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,987評論 3 395
  • 文/潘曉璐 我一進(jìn)店門敌厘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來台猴,“玉大人朽合,你說我怎么就攤上這事”タ瘢” “怎么了曹步?”我有些...
    開封第一講書人閱讀 164,298評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長休讳。 經(jīng)常有香客問我讲婚,道長,這世上最難降的妖魔是什么俊柔? 我笑而不...
    開封第一講書人閱讀 58,586評論 1 293
  • 正文 為了忘掉前任筹麸,我火速辦了婚禮,結(jié)果婚禮上雏婶,老公的妹妹穿的比我還像新娘物赶。我一直安慰自己,他們只是感情好留晚,可當(dāng)我...
    茶點故事閱讀 67,633評論 6 392
  • 文/花漫 我一把揭開白布酵紫。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪奖地。 梳的紋絲不亂的頭發(fā)上橄唬,一...
    開封第一講書人閱讀 51,488評論 1 302
  • 那天,我揣著相機與錄音参歹,去河邊找鬼仰楚。 笑死,一個胖子當(dāng)著我的面吹牛犬庇,可吹牛的內(nèi)容都是我干的缸血。 我是一名探鬼主播,決...
    沈念sama閱讀 40,275評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼械筛,長吁一口氣:“原來是場噩夢啊……” “哼捎泻!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起埋哟,我...
    開封第一講書人閱讀 39,176評論 0 276
  • 序言:老撾萬榮一對情侶失蹤笆豁,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后赤赊,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體闯狱,經(jīng)...
    沈念sama閱讀 45,619評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,819評論 3 336
  • 正文 我和宋清朗相戀三年抛计,在試婚紗的時候發(fā)現(xiàn)自己被綠了哄孤。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,932評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡吹截,死狀恐怖瘦陈,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情波俄,我是刑警寧澤晨逝,帶...
    沈念sama閱讀 35,655評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站懦铺,受9級特大地震影響捉貌,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜冬念,卻給世界環(huán)境...
    茶點故事閱讀 41,265評論 3 329
  • 文/蒙蒙 一趁窃、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧急前,春花似錦醒陆、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,871評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽检碗。三九已至,卻和暖如春码邻,著一層夾襖步出監(jiān)牢的瞬間折剃,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,994評論 1 269
  • 我被黑心中介騙來泰國打工像屋, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留怕犁,地道東北人。 一個月前我還...
    沈念sama閱讀 48,095評論 3 370
  • 正文 我出身青樓己莺,卻偏偏與公主長得像奏甫,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子凌受,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,884評論 2 354