iOS Crash日志 收集

今天在微信公眾號上看到一篇文章,做一下簡化整理珠漂,大家可以嘗試一起來做一下自己的Crash日志記錄

開發(fā)iOS應用崩溪,解決Crash問題始終是一個難題。Crash分為兩種溢谤,一種是由EXC_BAD_ACCESS引起的瞻凤,原因是訪問了不屬于本進程的內存地址憨攒,有可能是訪問已被釋放的內存;另一種是未被捕獲的Objective-C異常(NSException)阀参,導致程序向自身發(fā)送了SIGABRT信號而崩潰浓恶。其實對于未捕獲的Objective-C異常,我們是有辦法將它記錄下來的结笨,如果日志記錄得當包晰,能夠解決絕大部分崩潰的問題。

一. 系統(tǒng)Crash

對于系統(tǒng)Crash而引起的程序異常退出炕吸,可以通過UncaughtExceptionHandler機制捕獲伐憾;也就是說在程序中catch以外的內容,被系統(tǒng)自帶的錯誤處理而捕獲赫模。我們要做的就是用自定義的函數(shù)替代該ExceptionHandler即可树肃。

二. 處理signal

使用Objective-C的異常處理是不能得到signal的,如果要處理它瀑罗,我們還要利用unix標準的signal機制胸嘴,注冊SIGABRT, SIGBUS, SIGSEGV等信號發(fā)生時的處理函數(shù)。該函數(shù)中我們可以輸出棧信息斩祭,版本信息等其他一切我們所想要的劣像。

下面是一些信號說明:
1) SIGHUP
本信號在用戶終端連接(正常或非正常)結束時發(fā)出, 通常是在終端的控制進程結束時, 通知同一session內的各個作業(yè), 這時它們與控制終端不再關聯(lián)摧玫。
登錄Linux時耳奕,系統(tǒng)會分配給登錄用戶一個終端(Session)。在這個終端運行的所有程序诬像,包括前臺進程組和后臺進程組屋群,一般都屬于這個 Session。當用戶退出Linux登錄時坏挠,前臺進程組和后臺有對終端輸出的進程將會收到SIGHUP信號芍躏。這個信號的默認操作為終止進程,因此前臺進 程組和后臺有終端輸出的進程就會中止降狠。不過可以捕獲這個信號对竣,比如wget能捕獲SIGHUP信號,并忽略它喊熟,這樣就算退出了Linux登錄柏肪, wget也 能繼續(xù)下載。
此外芥牌,對于與終端脫離關系的守護進程烦味,這個信號用于通知它重新讀取配置文件。

2) SIGINT
程序終止(interrupt)信號, 在用戶鍵入INTR字符(通常是Ctrl-C)時發(fā)出,用于通知前臺進程組終止進程谬俄。

3) SIGQUIT
和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進程在因收到SIGQUIT退出時會產(chǎn)生core文件, 在這個意義上類似于一個程序錯誤信號柏靶。

4) SIGILL
執(zhí)行了非法指令. 通常是因為可執(zhí)行文件本身出現(xiàn)錯誤, 或者試圖執(zhí)行數(shù)據(jù)段. 堆棧溢出時也有可能產(chǎn)生這個信號。

5) SIGTRAP
由斷點指令或其它trap指令產(chǎn)生. 由debugger使用溃论。

6) SIGABRT
調用abort函數(shù)生成的信號屎蜓。

7) SIGBUS
非法地址, 包括內存地址對齊(alignment)出錯。比如訪問一個四個字長的整數(shù), 但其地址不是4的倍數(shù)钥勋。它與SIGSEGV的區(qū)別在于后者是由于對合法存儲地址的非法訪問觸發(fā)的(如訪問不屬于自己存儲空間或只讀存儲空間)炬转。

8) SIGFPE
在發(fā)生致命的算術運算錯誤時發(fā)出. 不僅包括浮點運算錯誤, 還包括溢出及除數(shù)為0等其它所有的算術的錯誤。

9) SIGKILL
用來立即結束程序的運行. 本信號不能被阻塞算灸、處理和忽略扼劈。如果管理員發(fā)現(xiàn)某個進程終止不了,可嘗試發(fā)送這個信號菲驴。

10) SIGUSR1
留給用戶使用

11) SIGSEGV
試圖訪問未分配給自己的內存, 或試圖往沒有寫權限的內存地址寫數(shù)據(jù).

12) SIGUSR2
留給用戶使用

13) SIGPIPE
管道破裂荐吵。這個信號通常在進程間通信產(chǎn)生,比如采用FIFO(管道)通信的兩個進程赊瞬,讀管道沒打開或者意外終止就往管道寫先煎,寫進程會收到SIGPIPE信號。此外用Socket通信的兩個進程巧涧,寫進程在寫Socket的時候薯蝎,讀進程已經(jīng)終止。

14) SIGALRM
時鐘定時信號, 計算的是實際的時間或時鐘時間. alarm函數(shù)使用該信號.

15) SIGTERM
程序結束(terminate)信號, 與SIGKILL不同的是該信號可以被阻塞和處理褒侧。通常用來要求程序自己正常退出良风,shell命令kill缺省產(chǎn)生這個信號。如果進程終止不了闷供,我們才會嘗試SIGKILL。

17) SIGCHLD
子進程結束時, 父進程會收到這個信號统诺。
如果父進程沒有處理這個信號歪脏,也沒有等待(wait)子進程,子進程雖然終止粮呢,但是還會在內核進程表中占有表項婿失,這時的子進程稱為僵尸進程。這種情 況我們應該避免(父進程或者忽略SIGCHILD信號啄寡,或者捕捉它豪硅,或者wait它派生的子進程,或者父進程先終止挺物,這時子進程的終止自動由init進程 來接管)懒浮。

18) SIGCONT
讓一個停止(stopped)的進程繼續(xù)執(zhí)行. 本信號不能被阻塞. 可以用一個handler來讓程序在由stopped狀態(tài)變?yōu)槔^續(xù)執(zhí)行時完成特定的工作. 例如, 重新顯示提示符

19) SIGSTOP
停止(stopped)進程的執(zhí)行. 注意它和terminate以及interrupt的區(qū)別:該進程還未結束, 只是暫停執(zhí)行. 本信號不能被阻塞, 處理或忽略.

20) SIGTSTP
停止進程的運行, 但該信號可以被處理和忽略. 用戶鍵入SUSP字符時(通常是Ctrl-Z)發(fā)出這個信號

21) SIGTTIN
當后臺作業(yè)要從用戶終端讀數(shù)據(jù)時, 該作業(yè)中的所有進程會收到SIGTTIN信號. 缺省時這些進程會停止執(zhí)行.

22) SIGTTOU
類似于SIGTTIN, 但在寫終端(或修改終端模式)時收到.

23) SIGURG
有”緊急”數(shù)據(jù)或out-of-band數(shù)據(jù)到達socket時產(chǎn)生.

24) SIGXCPU
超過CPU時間資源限制. 這個限制可以由getrlimit/setrlimit來讀取/改變。

25) SIGXFSZ
當進程企圖擴大文件以至于超過文件大小資源限制。

26) SIGVTALRM
虛擬時鐘信號. 類似于SIGALRM, 但是計算的是該進程占用的CPU時間.

27) SIGPROF
類似于SIGALRM/SIGVTALRM, 但包括該進程用的CPU時間以及系統(tǒng)調用的時間.

28) SIGWINCH
窗口大小改變時發(fā)出.

29) SIGIO
文件描述符準備就緒, 可以開始進行輸入/輸出操作.

30) SIGPWR
Power failure

31) SIGSYS
非法的系統(tǒng)調用砚著。

關鍵點
在以上列出的信號中次伶,程序不可捕獲、阻塞或忽略的信號有:SIGKILL,SIGSTOP
不能恢復至默認動作的信號有:SIGILL,SIGTRAP
默認會導致進程流產(chǎn)的信號有:SIGABRT,SIGBUS,SIGFPE,SIGILL,SIGIOT,SIGQUIT,SIGSEGV,SIGTRAP,SIGXCPU,SIGXFSZ
默認會導致進程退出的信號有:
SIGALRM,SIGHUP,SIGINT,SIGKILL,SIGPIPE,SIGPOLL,SIGPROF,SIGSYS,SIGTERM,SIGUSR1,SIGUSR2,SIGVTALRM
默認會導致進程停止的信號有:SIGSTOP,SIGTSTP,SIGTTIN,SIGTTOU
默認進程忽略的信號有:SIGCHLD,SIGPWR,SIGURG,SIGWINCH
此外稽穆,SIGIO在SVR4是退出冠王,在4.3BSD中是忽略;SIGCONT在進程掛起時是繼續(xù)舌镶,否則是忽略柱彻,不能被阻塞。
【具體原因為啥餐胀,我也不知道??哟楷,問微信大神去把】

好了,說了那么多骂澄,終于可以實戰(zhàn)寫代碼了:
1.AppDelegate.m中

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// Override point for customization after application launch.
    
InstallSignalHandler();//信號量截斷
InstallUncaughtExceptionHandler();//系統(tǒng)異常捕獲
    
return YES;
}

2.SignalHandler.m的實現(xiàn)

void SignalExceptionHandler(int signal)
{
    NSMutableString *mstr = [[NSMutableString alloc] init];
    [mstr appendString:@"Stack: "];
    void* callstack[128];
    int i, frames = backtrace(callstack, 128);
    char** strs = backtrace_symbols(callstack, frames);
    for (i = 0; i <frames; ++i) {
        [mstr appendFormat:@"%s ", strs[i]];
    }
    [SignalHandler saveCreash:mstr];
 
}

void InstallSignalHandler(void)
{
    signal(SIGHUP, SignalExceptionHandler);
    signal(SIGINT, SignalExceptionHandler);
    signal(SIGQUIT, SignalExceptionHandler);
    
    signal(SIGABRT, SignalExceptionHandler);
    signal(SIGILL, SignalExceptionHandler);
    signal(SIGSEGV, SignalExceptionHandler);
    signal(SIGFPE, SignalExceptionHandler);
    signal(SIGBUS, SignalExceptionHandler);
    signal(SIGPIPE, SignalExceptionHandler);
}

3.UncaughtExceptionHandler.m的實現(xiàn)

void HandleException(NSException *exception)
{
    // 異常的堆棧信息
    NSArray *stackArray = [exception callStackSymbols];
    // 出現(xiàn)異常的原因
    NSString *reason = [exception reason];
    // 異常名稱
    NSString *name = [exception name];
    NSString *exceptionInfo = [NSString stringWithFormat:@"Exception reason:%@ Exception name:%@ Exception stack:%@",name, reason, stackArray];
    NSLog(@"%@", exceptionInfo);
    [UncaughtExceptionHandler saveCreash:exceptionInfo];
}
 
void InstallUncaughtExceptionHandler(void)
{
    NSSetUncaughtExceptionHandler(&HandleException);
}

4吓蘑、測試一下
這里最關鍵的一步,SignalHandler不要在debug環(huán)境下測試坟冲。因為系統(tǒng)的debug會優(yōu)先去攔截磨镶。我們要運行一次后,關閉debug狀態(tài)健提。應該直接在模擬器上點擊我們build上去的app去運行琳猫。而UncaughtExceptionHandler可以在調試狀態(tài)下捕捉

- (void)buttonClick:(UIButton *)sender {
//1.信號量
    Test *pTest = {1,2};
    free(pTest);//導致SIGABRT的錯誤,因為內存中根本就沒有這個空間私痹,哪來的free脐嫂,就在棧中的對象而已
    pTest->a = 5;
}
- (IBAction)buttonOCException:(UIButton *)sender
{
    //2.ios崩潰
    NSArray *array= @[@"tom",@"xxx",@"ooo"];
    [array objectAtIndex:5];
}

附上demo:【demo下載地址

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市紊遵,隨后出現(xiàn)的幾起案子账千,更是在濱河造成了極大的恐慌,老刑警劉巖暗膜,帶你破解...
    沈念sama閱讀 222,000評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件匀奏,死亡現(xiàn)場離奇詭異,居然都是意外死亡学搜,警方通過查閱死者的電腦和手機娃善,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來瑞佩,“玉大人聚磺,你說我怎么就攤上這事【嫱瑁” “怎么了瘫寝?”我有些...
    開封第一講書人閱讀 168,561評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我矢沿,道長滥搭,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,782評論 1 298
  • 正文 為了忘掉前任捣鲸,我火速辦了婚禮瑟匆,結果婚禮上,老公的妹妹穿的比我還像新娘栽惶。我一直安慰自己愁溜,他們只是感情好,可當我...
    茶點故事閱讀 68,798評論 6 397
  • 文/花漫 我一把揭開白布外厂。 她就那樣靜靜地躺著冕象,像睡著了一般。 火紅的嫁衣襯著肌膚如雪汁蝶。 梳的紋絲不亂的頭發(fā)上渐扮,一...
    開封第一講書人閱讀 52,394評論 1 310
  • 那天,我揣著相機與錄音掖棉,去河邊找鬼墓律。 笑死,一個胖子當著我的面吹牛幔亥,可吹牛的內容都是我干的耻讽。 我是一名探鬼主播,決...
    沈念sama閱讀 40,952評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼帕棉,長吁一口氣:“原來是場噩夢啊……” “哼针肥!你這毒婦竟也來了?” 一聲冷哼從身側響起香伴,我...
    開封第一講書人閱讀 39,852評論 0 276
  • 序言:老撾萬榮一對情侶失蹤慰枕,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后即纲,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體捺僻,經(jīng)...
    沈念sama閱讀 46,409評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 38,483評論 3 341
  • 正文 我和宋清朗相戀三年崇裁,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片束昵。...
    茶點故事閱讀 40,615評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡拔稳,死狀恐怖,靈堂內的尸體忽然破棺而出锹雏,到底是詐尸還是另有隱情巴比,我是刑警寧澤,帶...
    沈念sama閱讀 36,303評論 5 350
  • 正文 年R本政府宣布,位于F島的核電站轻绞,受9級特大地震影響采记,放射性物質發(fā)生泄漏。R本人自食惡果不足惜政勃,卻給世界環(huán)境...
    茶點故事閱讀 41,979評論 3 334
  • 文/蒙蒙 一唧龄、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧奸远,春花似錦既棺、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,470評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至薛窥,卻和暖如春胖烛,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背诅迷。 一陣腳步聲響...
    開封第一講書人閱讀 33,571評論 1 272
  • 我被黑心中介騙來泰國打工佩番, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人竟贯。 一個月前我還...
    沈念sama閱讀 49,041評論 3 377
  • 正文 我出身青樓答捕,卻偏偏與公主長得像,于是被迫代替她去往敵國和親屑那。 傳聞我的和親對象是個殘疾皇子拱镐,可洞房花燭夜當晚...
    茶點故事閱讀 45,630評論 2 359

推薦閱讀更多精彩內容