iOS異常捕獲

轉(zhuǎn)載自:向晨宇的技術(shù)博客-iOS異常捕獲

文章目錄
前言
一. 系統(tǒng)Crash
二. 處理signal
1. 下面是一些信號(hào)說明
2. 關(guān)鍵點(diǎn)注意
三. 實(shí)戰(zhàn)
四. Crash Callstack分析 - 進(jìn)?一步分析
五. demo地址
六. 參考文獻(xiàn)

前言

今天在ios高級(jí)群237305299,有朋友問到iOS的異常捕捉的問題土辩,這一塊以前也沒有研究過禽额,趁此機(jī)會(huì)研究了一把钱反。并寫了一個(gè)demo楼眷,如有需要可以在文章最下面去下載飞主。

在閱讀文章之前,建議大家在閱讀完此篇文章后可以閱讀念茜 漫談iOS Crash收集框架,了解一下原理妙黍。

開發(fā)iOS應(yīng)用,解決Crash問題始終是一個(gè)難題瞧剖。Crash分為兩種拭嫁,一種是由EXC_BAD_ACCESS引起的,原因是訪問了不屬于本進(jìn)程的內(nèi)存地址抓于,有可能是訪問已被釋放的內(nèi)存做粤;另一種是未被捕獲的Objective-C異常(NSException),導(dǎo)致程序向自身發(fā)送了SIGABRT信號(hào)而崩潰毡咏。其實(shí)對(duì)于未捕獲的Objective-C異常驮宴,我們是有辦法將它記錄下來的,如果日志記錄得當(dāng)呕缭,能夠解決絕大部分崩潰的問題堵泽。這里對(duì)于UI線程與后臺(tái)線程分別說明

一. 系統(tǒng)Crash

對(duì)于系統(tǒng)Crash而引起的程序異常退出修己,可以通過UncaughtExceptionHandler機(jī)制捕獲;也就是說在程序中catch以外的內(nèi)容迎罗,被系統(tǒng)自帶的錯(cuò)誤處理而捕獲睬愤。我們要做的就是用自定義的函數(shù)替代該ExceptionHandler即可。

二. 處理signal

使用Objective-C的異常處理是不能得到signal的纹安,如果要處理它尤辱,我們還要利用unix標(biāo)準(zhǔn)的signal機(jī)制,注冊(cè)SIGABRT, SIGBUS, SIGSEGV等信號(hào)發(fā)生時(shí)的處理函數(shù)厢岂。該函數(shù)中我們可以輸出棧信息光督,版本信息等其他一切我們所想要的。

下面是一些信號(hào)說明
  1. SIGHUP
    本信號(hào)在用戶終端連接(正乘#或非正常)結(jié)束時(shí)發(fā)出, 通常是在終端的控制進(jìn)程結(jié)束時(shí), 通知同一session內(nèi)的各個(gè)作業(yè), 這時(shí)它們與控制終端不再關(guān)聯(lián)结借。
    登錄Linux時(shí),系統(tǒng)會(huì)分配給登錄用戶一個(gè)終端(Session)卒茬。在這個(gè)終端運(yùn)行的所有程序船老,包括前臺(tái)進(jìn)程組和后臺(tái)進(jìn)程組,一般都屬于這個(gè) Session圃酵。當(dāng)用戶退出Linux登錄時(shí)柳畔,前臺(tái)進(jìn)程組和后臺(tái)有對(duì)終端輸出的進(jìn)程將會(huì)收到SIGHUP信號(hào)。這個(gè)信號(hào)的默認(rèn)操作為終止進(jìn)程郭赐,因此前臺(tái)進(jìn) 程組和后臺(tái)有終端輸出的進(jìn)程就會(huì)中止薪韩。不過可以捕獲這個(gè)信號(hào),比如wget能捕獲SIGHUP信號(hào)堪置,并忽略它躬存,這樣就算退出了Linux登錄张惹, wget也 能繼續(xù)下載舀锨。
    此外,對(duì)于與終端脫離關(guān)系的守護(hù)進(jìn)程宛逗,這個(gè)信號(hào)用于通知它重新讀取配置文件坎匿。
  2. SIGINT
    程序終止(interrupt)信號(hào), 在用戶鍵入INTR字符(通常是Ctrl-C)時(shí)發(fā)出,用于通知前臺(tái)進(jìn)程組終止進(jìn)程雷激。
  3. SIGQUIT
    和SIGINT類似, 但由QUIT字符(通常是Ctrl-)來控制. 進(jìn)程在因收到SIGQUIT退出時(shí)會(huì)產(chǎn)生core文件, 在這個(gè)意義上類似于一個(gè)程序錯(cuò)誤信號(hào)替蔬。
  4. SIGILL
    執(zhí)行了非法指令. 通常是因?yàn)榭蓤?zhí)行文件本身出現(xiàn)錯(cuò)誤, 或者試圖執(zhí)行數(shù)據(jù)段. 堆棧溢出時(shí)也有可能產(chǎn)生這個(gè)信號(hào)。
  5. SIGTRAP
    由斷點(diǎn)指令或其它trap指令產(chǎn)生. 由debugger使用屎暇。
  6. SIGABRT
    調(diào)用abort函數(shù)生成的信號(hào)承桥。
  7. SIGBUS
    非法地址, 包括內(nèi)存地址對(duì)齊(alignment)出錯(cuò)。比如訪問一個(gè)四個(gè)字長(zhǎng)的整數(shù), 但其地址不是4的倍數(shù)根悼。它與SIGSEGV的區(qū)別在于后者是由于對(duì)合法存儲(chǔ)地址的非法訪問觸發(fā)的(如訪問不屬于自己存儲(chǔ)空間或只讀存儲(chǔ)空間)凶异。
  8. SIGFPE
    在發(fā)生致命的算術(shù)運(yùn)算錯(cuò)誤時(shí)發(fā)出. 不僅包括浮點(diǎn)運(yùn)算錯(cuò)誤, 還包括溢出及除數(shù)為0等其它所有的算術(shù)的錯(cuò)誤蜀撑。
  9. SIGKILL
    用來立即結(jié)束程序的運(yùn)行. 本信號(hào)不能被阻塞、處理和忽略剩彬。如果管理員發(fā)現(xiàn)某個(gè)進(jìn)程終止不了酷麦,可嘗試發(fā)送這個(gè)信號(hào)。
  10. SIGUSR1
    留給用戶使用
  11. SIGSEGV
    試圖訪問未分配給自己的內(nèi)存, 或試圖往沒有寫權(quán)限的內(nèi)存地址寫數(shù)據(jù).
  12. SIGUSR2
    留給用戶使用
  13. SIGPIPE
    管道破裂喉恋。這個(gè)信號(hào)通常在進(jìn)程間通信產(chǎn)生沃饶,比如采用FIFO(管道)通信的兩個(gè)進(jìn)程,讀管道沒打開或者意外終止就往管道寫轻黑,寫進(jìn)程會(huì)收到SIGPIPE信號(hào)糊肤。此外用Socket通信的兩個(gè)進(jìn)程,寫進(jìn)程在寫Socket的時(shí)候氓鄙,讀進(jìn)程已經(jīng)終止轩褐。
  14. SIGALRM
    時(shí)鐘定時(shí)信號(hào), 計(jì)算的是實(shí)際的時(shí)間或時(shí)鐘時(shí)間. alarm函數(shù)使用該信號(hào).
  15. SIGTERM
    程序結(jié)束(terminate)信號(hào), 與SIGKILL不同的是該信號(hào)可以被阻塞和處理。通常用來要求程序自己正常退出玖详,shell命令kill缺省產(chǎn)生這個(gè)信號(hào)把介。如果進(jìn)程終止不了,我們才會(huì)嘗試SIGKILL蟋座。
  16. SIGCHLD
    子進(jìn)程結(jié)束時(shí), 父進(jìn)程會(huì)收到這個(gè)信號(hào)拗踢。
    如果父進(jìn)程沒有處理這個(gè)信號(hào),也沒有等待(wait)子進(jìn)程向臀,子進(jìn)程雖然終止巢墅,但是還會(huì)在內(nèi)核進(jìn)程表中占有表項(xiàng),這時(shí)的子進(jìn)程稱為僵尸進(jìn)程券膀。這種情 況我們應(yīng)該避免(父進(jìn)程或者忽略SIGCHILD信號(hào)君纫,或者捕捉它,或者wait它派生的子進(jìn)程芹彬,或者父進(jìn)程先終止蓄髓,這時(shí)子進(jìn)程的終止自動(dòng)由init進(jìn)程 來接管)。
  17. SIGCONT
    讓一個(gè)停止(stopped)的進(jìn)程繼續(xù)執(zhí)行. 本信號(hào)不能被阻塞. 可以用一個(gè)handler來讓程序在由stopped狀態(tài)變?yōu)槔^續(xù)執(zhí)行時(shí)完成特定的工作. 例如, 重新顯示提示符
  18. SIGSTOP
    停止(stopped)進(jìn)程的執(zhí)行. 注意它和terminate以及interrupt的區(qū)別:該進(jìn)程還未結(jié)束, 只是暫停執(zhí)行. 本信號(hào)不能被阻塞, 處理或忽略.
  19. SIGTSTP
    停止進(jìn)程的運(yùn)行, 但該信號(hào)可以被處理和忽略. 用戶鍵入SUSP字符時(shí)(通常是Ctrl-Z)發(fā)出這個(gè)信號(hào)
  20. SIGTTIN
    當(dāng)后臺(tái)作業(yè)要從用戶終端讀數(shù)據(jù)時(shí), 該作業(yè)中的所有進(jìn)程會(huì)收到SIGTTIN信號(hào). 缺省時(shí)這些進(jìn)程會(huì)停止執(zhí)行.
  21. SIGTTOU
    類似于SIGTTIN, 但在寫終端(或修改終端模式)時(shí)收到.
  22. SIGURG
    有”緊急”數(shù)據(jù)或out-of-band數(shù)據(jù)到達(dá)socket時(shí)產(chǎn)生.
  23. SIGXCPU
    超過CPU時(shí)間資源限制. 這個(gè)限制可以由getrlimit/setrlimit來讀取/改變舒帮。
  24. SIGXFSZ
    當(dāng)進(jìn)程企圖擴(kuò)大文件以至于超過文件大小資源限制会喝。
  25. SIGVTALRM
    虛擬時(shí)鐘信號(hào). 類似于SIGALRM, 但是計(jì)算的是該進(jìn)程占用的CPU時(shí)間.
  26. SIGPROF
    類似于SIGALRM/SIGVTALRM, 但包括該進(jìn)程用的CPU時(shí)間以及系統(tǒng)調(diào)用的時(shí)間.
  27. SIGWINCH
    窗口大小改變時(shí)發(fā)出.
  28. SIGIO
    文件描述符準(zhǔn)備就緒, 可以開始進(jìn)行輸入/輸出操作.
  29. SIGPWR
    Power failure
  30. SIGSYS
    非法的系統(tǒng)調(diào)用。
關(guān)鍵點(diǎn)注意
  • 在以上列出的信號(hào)中玩郊,程序不可捕獲肢执、阻塞或忽略的信號(hào)有:SIGKILL,SIGSTOP
  • 不能恢復(fù)至默認(rèn)動(dòng)作的信號(hào)有:SIGILL,SIGTRAP
  • 默認(rèn)會(huì)導(dǎo)致進(jìn)程流產(chǎn)的信號(hào)有:SIGABRT,SIGBUS,SIGFPE,SIGILL,SIGIOT,SIGQUIT,SIGSEGV,SIGTRAP,SIGXCPU,SIGXFSZ
    默認(rèn)會(huì)導(dǎo)致進(jìn)程退出的信號(hào)有:
  • SIGALRM,SIGHUP,SIGINT,SIGKILL,SIGPIPE,SIGPOLL,SIGPROF,SIGSYS,SIGTERM,SIGUSR1,SIGUSR2,SIGVTALRM
  • 默認(rèn)會(huì)導(dǎo)致進(jìn)程停止的信號(hào)有:SIGSTOP,SIGTSTP,SIGTTIN,SIGTTOU
  • 默認(rèn)進(jìn)程忽略的信號(hào)有:SIGCHLD,SIGPWR,SIGURG,SIGWINCH
  • 此外,SIGIO在SVR4是退出译红,在4.3BSD中是忽略预茄;SIGCONT在進(jìn)程掛起時(shí)是繼續(xù),否則是忽略侦厚,不能被阻塞耻陕。

三. 實(shí)戰(zhàn)

1.AppDelegate.m

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.

    InstallSignalHandler();//信號(hào)量截?cái)?    InstallUncaughtExceptionHandler();//系統(tǒng)異常捕獲

    return YES;
}

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

void SignalExceptionHandler(int signal)
{
   NSMutableString *mstr = [[NSMutableString alloc] init];
   [mstr appendString:@"Stack:\n"];
   void* callstack[128];
   int i, frames = backtrace(callstack, 128);
   char** strs = backtrace_symbols(callstack, frames);
   for (i = 0; i <frames; ++i) {
       [mstr appendFormat:@"%s\n", 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);
}

有關(guān)錯(cuò)誤類型可以看上面的說明昵慌,SignalExceptionHandler是信號(hào)出錯(cuò)時(shí)候的回調(diào)。當(dāng)有信號(hào)出錯(cuò)的時(shí)候淮蜈,可以回調(diào)到這個(gè)方法

3.UncaughtExceptionHandler.m的實(shí)現(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:%@\nException 
   name:%@\nException stack:%@",name, reason, stackArray];
   NSLog(@"%@", exceptionInfo);
   [UncaughtExceptionHandler saveCreash:exceptionInfo];
}

void InstallUncaughtExceptionHandler(void)
{
   NSSetUncaughtExceptionHandler(&HandleException);
}

4.測(cè)試–踩坑關(guān)鍵
這里最關(guān)鍵的一步斋攀,SignalHandler不要在debug環(huán)境下測(cè)試。因?yàn)橄到y(tǒng)的debug會(huì)優(yōu)先去攔截梧田。我們要運(yùn)行一次后淳蔼,關(guān)閉debug狀態(tài)。應(yīng)該直接在模擬器上點(diǎn)擊我們build上去的app去運(yùn)行裁眯。而UncaughtExceptionHandler可以在調(diào)試狀態(tài)下捕捉

- (IBAction)buttonClick:(UIButton *)sender {
    //1.信號(hào)量
   Test *pTest = {1,2};
   free(pTest);//導(dǎo)致SIGABRT的錯(cuò)誤鹉梨,因?yàn)閮?nèi)存中根本就沒有這個(gè)空間,哪來的free穿稳,就在棧中的對(duì)象而已
   pTest->a = 5;
}
- (IBAction)buttonOCException:(UIButton *)sender
{
   //2.ios崩潰
   NSArray *array= @[@"tom",@"xxx",@"ooo"];
   [array objectAtIndex:5];
}

"文件結(jié)構(gòu)"
"操作"
"操作"

四. Crash Callstack分析 - 進(jìn)?一步分析

屬性 說明
0x8badf00d 在啟動(dòng)存皂、終?止應(yīng)?用或響應(yīng)系統(tǒng)事件花費(fèi)過?長(zhǎng)時(shí)間,意為“ate bad food”。
0xdeadfa11 ?用戶強(qiáng)制退出,意為“dead fall”逢艘。(系統(tǒng)?無(wú)響應(yīng)時(shí),?用戶按電源開關(guān)和HOME)
0xbaaaaaad ?用戶按住Home鍵和?音量鍵,獲取當(dāng)前內(nèi)存狀態(tài),不代表崩潰
0xbad22222 VoIP應(yīng)?用因?yàn)榛謴?fù)得太頻繁導(dǎo)致crash
0xc00010ff 因?yàn)樘珷C了被干掉,意為“cool off”
0xdead10cc 因?yàn)樵诤笈_(tái)時(shí)仍然占據(jù)系統(tǒng)資源(?比如通訊錄)被干掉,意為“dead lock”
五. demo地址

iOSCrashUncaught下載

六. 參考文獻(xiàn)

1.程序crash后的調(diào)試技巧
2.iOS開發(fā)socket程序被SIGPIPE信號(hào)Terminate的問題
3.美女念茜
4.如何定位Obj-C野指針隨機(jī)Crash(一):先提高野指針Crash率
5.如何定位Obj-C野指針隨機(jī)Crash(二):讓非必現(xiàn)Crash變成必現(xiàn)
6.如何定位Obj-C野指針隨機(jī)Crash(三):加點(diǎn)黑科技讓Crash自報(bào)家門

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末旦袋,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子它改,更是在濱河造成了極大的恐慌疤孕,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,820評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件央拖,死亡現(xiàn)場(chǎng)離奇詭異祭阀,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)鲜戒,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,648評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門专控,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人遏餐,你說我怎么就攤上這事伦腐。” “怎么了境输?”我有些...
    開封第一講書人閱讀 168,324評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵蔗牡,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我嗅剖,道長(zhǎng),這世上最難降的妖魔是什么嘁扼? 我笑而不...
    開封第一講書人閱讀 59,714評(píng)論 1 297
  • 正文 為了忘掉前任信粮,我火速辦了婚禮,結(jié)果婚禮上趁啸,老公的妹妹穿的比我還像新娘强缘。我一直安慰自己督惰,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,724評(píng)論 6 397
  • 文/花漫 我一把揭開白布旅掂。 她就那樣靜靜地躺著赏胚,像睡著了一般。 火紅的嫁衣襯著肌膚如雪商虐。 梳的紋絲不亂的頭發(fā)上觉阅,一...
    開封第一講書人閱讀 52,328評(píng)論 1 310
  • 那天,我揣著相機(jī)與錄音秘车,去河邊找鬼典勇。 笑死,一個(gè)胖子當(dāng)著我的面吹牛叮趴,可吹牛的內(nèi)容都是我干的割笙。 我是一名探鬼主播,決...
    沈念sama閱讀 40,897評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼眯亦,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼伤溉!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起妻率,我...
    開封第一講書人閱讀 39,804評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤谈火,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后舌涨,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體糯耍,經(jīng)...
    沈念sama閱讀 46,345評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,431評(píng)論 3 340
  • 正文 我和宋清朗相戀三年囊嘉,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了温技。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,561評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡扭粱,死狀恐怖舵鳞,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情琢蛤,我是刑警寧澤蜓堕,帶...
    沈念sama閱讀 36,238評(píng)論 5 350
  • 正文 年R本政府宣布,位于F島的核電站博其,受9級(jí)特大地震影響套才,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜慕淡,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,928評(píng)論 3 334
  • 文/蒙蒙 一背伴、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦傻寂、人聲如沸息尺。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,417評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)搂誉。三九已至,卻和暖如春静檬,著一層夾襖步出監(jiān)牢的瞬間炭懊,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,528評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工巴柿, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留凛虽,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,983評(píng)論 3 376
  • 正文 我出身青樓广恢,卻偏偏與公主長(zhǎng)得像凯旋,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子钉迷,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,573評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • 文章目錄 一. 系統(tǒng)Crash 二. 處理signal 下面是一些信號(hào)說明 關(guān)鍵點(diǎn)注意 三. 實(shí)戰(zhàn) 四. Cras...
    MTDeveloper閱讀 1,161評(píng)論 1 2
  • 異常分類 軟件異常(OC異常) 主要來源于 kill() 至非、 pthread_kill() 兩個(gè) API 的調(diào)用,...
    conowen閱讀 2,210評(píng)論 1 1
  • 開發(fā)iOS應(yīng)用,解決Crash問題始終是一個(gè)難題糠聪。Crash分為兩種荒椭,一種是由EXC_BAD_ACCESS引起的,...
    流星Meteor閱讀 4,993評(píng)論 0 10
  • 今天在微信公眾號(hào)上看到一篇文章舰蟆,做一下簡(jiǎn)化整理趣惠,大家可以嘗試一起來做一下自己的Crash日志記錄 開發(fā)iOS應(yīng)用,...
    東健FO_OF閱讀 2,879評(píng)論 1 24
  • 突然感覺到現(xiàn)在的日子過的像白開水一樣無(wú)味平淡身害,雖說白開水無(wú)味確很健康味悄,但我總覺得生活還是得有點(diǎn)滋味。通俗的說就...
    努力奔跑的孩子閱讀 184評(píng)論 1 1