iOS:app崩潰怎么辦悍汛?崩潰捕獲

iOS中App崩潰問題是在所難免的,那如何在崩潰之前給用戶一些提示至会,或是可以將這些崩潰信息發(fā)送給自己离咐,以便于自己修正呢?前段時間做項目時奋献,一直沒添加這方面的功能健霹,這兩天沒事從網(wǎng)上看了些資料,順便自己做了個Demo測試了一下瓶蚂,今天把我了解的東西整理一下糖埋。

IOS SDK中提供了一個現(xiàn)成的函數(shù)NSSetUncaughtExceptionHandler 用來做異常處理,但功能非常有限窃这,而引起崩潰的大多數(shù)原因如:內(nèi)存訪問錯誤瞳别,重復(fù)釋放等錯誤就無能為力了,因為這種錯誤它拋出的是Signal杭攻,所以必須要專門做Signal處理祟敛。

Demo運行環(huán)境:Xcode7.3, iOS9.3

步驟一:定義一個UncaughtExceptionHandler類

UncaughtExceptionHandler.h內(nèi)容:

#import <UIKit/UIKit.h>

@interface UncaughtExceptionHandler : NSObject
{
  BOOL dismissed;
}

@end

void InstallUncaughtExceptionHandler();

UncaughtExceptionHandler.m內(nèi)容:

#import "UncaughtExceptionHandler.h"
#include <libkern/OSAtomic.h>
#include <execinfo.h>

NSString * const UncaughtExceptionHandlerSignalExceptionName = @"UncaughtExceptionHandlerSignalExceptionName";
NSString * const UncaughtExceptionHandlerSignalKey = @"UncaughtExceptionHandlerSignalKey";
NSString * const UncaughtExceptionHandlerAddressesKey = @"UncaughtExceptionHandlerAddressesKey";
volatile int32_t UncaughtExceptionCount = 0;
const int32_t UncaughtExceptionMaximum = 10;
const NSInteger UncaughtExceptionHandlerSkipAddressCount = 4;
const NSInteger UncaughtExceptionHandlerReportAddressCount = 5;

@implementation UncaughtExceptionHandler

+ (NSArray *)backtrace
{
    void* callstack[128];
    int frames = backtrace(callstack, 128);
    char **strs = backtrace_symbols(callstack, frames);
    
    int i;
    NSMutableArray *backtrace = [NSMutableArray arrayWithCapacity:frames];
    for (
         i = UncaughtExceptionHandlerSkipAddressCount;
         i < UncaughtExceptionHandlerSkipAddressCount + UncaughtExceptionHandlerReportAddressCount;
         i++
         )
    {
        [backtrace addObject:[NSString stringWithUTF8String:strs[i]]];
    }
    free(strs);
    
    return backtrace;
}

- (void)alertView:(UIAlertView *)anAlertView clickedButtonAtIndex:(NSInteger)anIndex
{
    if (anIndex == 0)
    {
        dismissed = YES;
    }
}

- (void)handleException:(NSException *)exception
{
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"程序出現(xiàn)問題啦", nil) message:[NSString stringWithFormat:NSLocalizedString(@"崩潰信息\n"@"%@\n%@", nil), [exception reason], [[exception userInfo] objectForKey:UncaughtExceptionHandlerAddressesKey]] delegate:self cancelButtonTitle:NSLocalizedString(@"離開", nil) otherButtonTitles:nil];
    [alert show];
    CFRunLoopRef runLoop = CFRunLoopGetCurrent();
    CFArrayRef allModes = CFRunLoopCopyAllModes(runLoop);
    while (!dismissed)
    {
        for (NSString *mode in (__bridge NSArray *)allModes)
        {
            CFRunLoopRunInMode((CFStringRef)mode, 0.001, false);
        }
    }
    CFRelease(allModes);
    NSSetUncaughtExceptionHandler(NULL);
    signal(SIGABRT, SIG_DFL);
    signal(SIGILL, SIG_DFL);
    signal(SIGSEGV, SIG_DFL);
    signal(SIGFPE, SIG_DFL);
    signal(SIGBUS, SIG_DFL);
    signal(SIGPIPE, SIG_DFL);
    if ([[exception name] isEqual:UncaughtExceptionHandlerSignalExceptionName])
    {
        kill(getpid(), [[[exception userInfo] objectForKey:UncaughtExceptionHandlerSignalKey] intValue]);
    }
    else
    {
        [exception raise];
    }
}
@end


NSString* getAppInfo()
{
    NSString *appInfo = [NSString stringWithFormat:@"App : %@ %@(%@)\nDevice : %@\nOS Version : %@ %@\n",
                         [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleDisplayName"],
                         [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleShortVersionString"],
                         [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleVersion"],
                         [UIDevice currentDevice].model,
                         [UIDevice currentDevice].systemName,
                         [UIDevice currentDevice].systemVersion];
    NSLog(@"Crash!!!! %@", appInfo);
    return appInfo;
}

void MySignalHandler(int signal)
{
    int32_t exceptionCount = OSAtomicIncrement32(&UncaughtExceptionCount);
    if (exceptionCount > UncaughtExceptionMaximum)
    {
        return;
    }
    NSMutableDictionary *userInfo = [NSMutableDictionary dictionaryWithObject:[NSNumber numberWithInt:signal] forKey:UncaughtExceptionHandlerSignalKey];
    NSArray *callStack = [UncaughtExceptionHandler backtrace];
    [userInfo setObject:callStack forKey:UncaughtExceptionHandlerAddressesKey];
    // 創(chuàng)建一個OC異常對象
    NSException *ex = [NSException exceptionWithName:UncaughtExceptionHandlerSignalExceptionName reason:[NSString stringWithFormat:NSLocalizedString(@"Signal %d was raised.\n"@"%@", nil), signal, getAppInfo()] userInfo:userInfo];
    // 處理異常消息
    [[[UncaughtExceptionHandler alloc] init] performSelectorOnMainThread:@selector(handleException:) withObject:ex waitUntilDone:YES];
}

void InstallUncaughtExceptionHandler()
{
    signal(SIGABRT, MySignalHandler);
    signal(SIGILL, MySignalHandler);
    signal(SIGSEGV, MySignalHandler);
    signal(SIGFPE, MySignalHandler);
    signal(SIGBUS, MySignalHandler);
    signal(SIGPIPE, MySignalHandler);
}
步驟二:在AppDelegate中導(dǎo)入本類
#import "UncaughtExceptionHandler.h"

- (void)installUncaughtExceptionHandler
{
    InstallUncaughtExceptionHandler();
}

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    [self installUncaughtExceptionHandler];
    return YES;
}
效果圖如下:
崩潰截圖.png

當(dāng)然,也可以把這些崩潰信息通過郵箱發(fā)送給自己兆解,這個根據(jù)項目需求來自定義馆铁。

[參考]http://www.111cn.net/sj/ios8/89772.htm

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市锅睛,隨后出現(xiàn)的幾起案子埠巨,更是在濱河造成了極大的恐慌,老刑警劉巖现拒,帶你破解...
    沈念sama閱讀 217,406評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件往枣,死亡現(xiàn)場離奇詭異衷掷,居然都是意外死亡泣特,警方通過查閱死者的電腦和手機(jī)基显,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評論 3 393
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人例驹,你說我怎么就攤上這事捐韩。” “怎么了鹃锈?”我有些...
    開封第一講書人閱讀 163,711評論 0 353
  • 文/不壞的土叔 我叫張陵奥帘,是天一觀的道長。 經(jīng)常有香客問我仪召,道長寨蹋,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,380評論 1 293
  • 正文 為了忘掉前任扔茅,我火速辦了婚禮已旧,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘召娜。我一直安慰自己运褪,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,432評論 6 392
  • 文/花漫 我一把揭開白布玖瘸。 她就那樣靜靜地躺著秸讹,像睡著了一般。 火紅的嫁衣襯著肌膚如雪雅倒。 梳的紋絲不亂的頭發(fā)上璃诀,一...
    開封第一講書人閱讀 51,301評論 1 301
  • 那天,我揣著相機(jī)與錄音蔑匣,去河邊找鬼劣欢。 笑死,一個胖子當(dāng)著我的面吹牛裁良,可吹牛的內(nèi)容都是我干的凿将。 我是一名探鬼主播,決...
    沈念sama閱讀 40,145評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼价脾,長吁一口氣:“原來是場噩夢啊……” “哼牧抵!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起侨把,我...
    開封第一講書人閱讀 39,008評論 0 276
  • 序言:老撾萬榮一對情侶失蹤犀变,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后座硕,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體弛作,經(jīng)...
    沈念sama閱讀 45,443評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡涕蜂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,649評論 3 334
  • 正文 我和宋清朗相戀三年华匾,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,795評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡蜘拉,死狀恐怖萨西,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情旭旭,我是刑警寧澤谎脯,帶...
    沈念sama閱讀 35,501評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站持寄,受9級特大地震影響源梭,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜稍味,卻給世界環(huán)境...
    茶點故事閱讀 41,119評論 3 328
  • 文/蒙蒙 一废麻、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧模庐,春花似錦烛愧、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至疼燥,卻和暖如春沧卢,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背醉者。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評論 1 269
  • 我被黑心中介騙來泰國打工搏恤, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人湃交。 一個月前我還...
    沈念sama閱讀 47,899評論 2 370
  • 正文 我出身青樓熟空,卻偏偏與公主長得像,于是被迫代替她去往敵國和親搞莺。 傳聞我的和親對象是個殘疾皇子息罗,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,724評論 2 354

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

  • 哲學(xué)是智慧的學(xué)問,思考著人類終極的問題才沧,對影響著人類未來命運的教育有著深刻的意義迈喉。 與西方哲學(xué)對教...
    何泉_1ec8閱讀 649評論 0 0
  • 文/淺商商 人,墜落凡間沒有腳的鳥 只能不斷的飛來飛去 沒有風(fēng)的日子 看著云守候雨 沒有過夢的日子 看著歲月劃過的...
    尤下閱讀 808評論 0 2
  • 煙花三月温圆,癡情者眾挨摸,是花癡迷情的季節(jié),酒醉情迷岁歉,那顆放蕩不羈的心隨著花開怦然而動得运!剩養(yǎng)病之閑留下些許文字。...
    人間未了情閱讀 373評論 0 2