强大的NSException

一、NSException简介

1.什么是NSException?

说到NSException你可能不太了解,但是下面的这张图你肯定见过不止一次


exception

这些就是NSException产生的,一旦程序抛出异常,程序就会崩溃,控制台就会有这些崩溃日志。

2、定义一个NSException对象并抛出

NSString*name =@"exception name";

NSString*reason =@"exception reason";

NSException*exception = [NSException exceptionWithName:name reason:reason userInfo:nil];

@throw exception;

运行程序会发现输出和上图类似的日志:

exception

二、异常的简单处理

如果你在开发过程中可能对某段代码不信任,也就是有崩溃的可能,而在线上直接报错又会影响又会体验,那你就可以通过如下的方法处理:

NSMutableArray*array = [NSMutableArray array];

NSString*nilStr =nil;

@try{

//有可能会出现异常的代码,这里写的代码一定会出现问题

[array insertObject:nilStr atIndex:0];

}@catch(NSException *exception) {

//如果@try中的代码出现异常,就会执行这里的代码,也就可以在这里进行相应操作

NSLog(@"exception.name=%@,exception.reason=%@",exception.name,exception.reason);

//如果想要抛出异常就执行如下代码,程序就会崩溃,便于调试

// @throw exception;

}@finally{

//这里的代码一定会执行

}

三、防止潜在的崩溃风险

如果你并不知道程序运行到哪里会出现异常,或者说对于Foundation框架里有非常多常用的方法有导致崩溃的潜在危险,那么该如何拦截潜在的异常风险,并进行相应的处理,防止崩溃的出现呢?

解决办法是:

1.利用iOS的runtime特性和catalog添加新方法,替换掉系统的存在异常风险的方法。

2.利用异常捕获防止程序崩溃,并进行相应处理。

具体的步骤:

1.创建一个HandleCrash类:

TXHandleCrash.h

#import

#define HandleCrashLogBegin @"==========================handleCrashLogBegin==========================="

#define HandleCrashLogEnd @"=============================handleCrash==============================="

#define HandleCrashLogNotification @"HandleCrashLogNotification"

@interfaceTXHandleCrash :NSObject

/**

start handle crash

*/

+(void)startHandle;

/**

exchange class method

*/

+(void)handleClass:(Class)anClass exchangeClassMethod:(SEL)method1 Method:(SEL)method2;

/**

exchange instance method

*/

+(void)handleClass:(Class)anClass exchangeCInstanceMethod:(SEL)method1 Method:(SEL)method2;

/**

handle exception

@param remark remark

*/

+(void)handleException:(NSException*)exception remark:(NSString*)remark;

@end

TXHandleCrash.m

#import"TXHandleCrash.h"

#import

#import"NSDictionary+TXHandleCrash.h"

@implementationTXHandleCrash

+(void)startHandle

{

dispatch_once_ttoken;

dispatch_once(&token , ^{

[NSDictionaryhandleCrash];

});

}

+(void)handleClass:(Class)anClass exchangeClassMethod:(SEL)method1 Method:(SEL)method2

{

Methodmtd1 =class_getClassMethod(anClass, method1);

Methodmtd2 =class_getClassMethod(anClass, method2);

method_exchangeImplementations(mtd1, mtd2);

}

+(void)handleClass:(Class)anClass exchangeCInstanceMethod:(SEL)method1 Method:(SEL)method2

{

Methodmtd1 =class_getInstanceMethod(anClass, method1);

Methodmtd2 =class_getInstanceMethod(anClass, method2);

method_exchangeImplementations(mtd1, mtd2);

}

+(void)handleException:(NSException*)exception remark:(NSString*)remark

{

//堆栈数据

NSArray*callStackSymbols = [NSThreadcallStackSymbols];

//获取在哪个类的哪个方法中实例化的数组,并格式化:-[类名方法名]、+[类名方法名]

NSString*locationMsg = [selflocationExcptionThroughCallStackSymbols:callStackSymbols];

if(!locationMsg) {

locationMsg =@"崩溃位置定位失败,请查看函数调用栈排查错误";

}

NSString*exceptionName = exception.name;

NSString*exceptionReason = exception.reason;

NSString*exceptionLocation = [NSStringstringWithFormat:@"exception location:%@",locationMsg];

NSString*exceptionMsg = [NSStringstringWithFormat:@"\n\n%@\n\n%@\n%@\n%@\n%@\n\n%@\n\n",HandleCrashLogBegin, exceptionName, exceptionReason, exceptionLocation, remark,HandleCrashLogEnd];

NSLog(@"%@", exceptionMsg);

NSDictionary*exceptionInfoDic =@{

@"exceptionName": exceptionName,

@"exceptionReason": exceptionReason,

@"exceptionLocation": exceptionLocation,

@"remark": remark,

@"exception": exception,

@"callStackSymbols": callStackSymbols

};

//将错误信息放在字典里,用通知的形式发送出去

[[NSNotificationCenterdefaultCenter]postNotificationName:HandleCrashLogNotificationobject:niluserInfo:exceptionInfoDic];

}

+(NSString*)locationExcptionThroughCallStackSymbols:(NSArray*)callStackSymbols

{

__blockNSString*locationMsg =nil;

NSLog(@"callStackSymbols=%@",callStackSymbols);

//通过正则匹配出的格式为,-[类名方法名]、+[类名方法名]

NSString*regularExpStr =@"[-\\+]\\[.+\\]";

NSRegularExpression*regularExp = [[NSRegularExpressionalloc]initWithPattern:regularExpStroptions:NSRegularExpressionCaseInsensitiveerror:nil];

for(intindex =2; index

NSString*callStackSymbol = callStackSymbols[index];

[regularExpenumerateMatchesInString:callStackSymboloptions:NSMatchingReportProgressrange:NSMakeRange(0, callStackSymbol.length)usingBlock:^(NSTextCheckingResult*_Nullableresult,NSMatchingFlagsflags,BOOL*_Nonnullstop) {

if(result) {

NSString*tmpLocationMsg = [callStackSymbolsubstringWithRange:result.range];

//get class name

NSString*className = [tmpLocationMsgcomponentsSeparatedByString:@" "].firstObject;

className = [classNamecomponentsSeparatedByString:@"["].lastObject;

NSBundle*bundle = [NSBundlebundleForClass:NSClassFromString(className)];

//filter catalog and system Class

if(![classNamehasPrefix:@")"] && bundle == [NSBundlemainBundle]) {

locationMsg = tmpLocationMsg ;

}

*stop =YES;

}

}];

if(locationMsg.length) {

break;

}

}

returnlocationMsg ;

}

@end

2.创建一个NSDictionary的catalog

NSDictionary+TXHandleCrash.h

#import

@interfaceNSDictionary (TXHandleCrash)

+(void)handleCrash;

@end

NSDictionary+TXHandleCrash.m

#import"NSDictionary+TXHandleCrash.h"

#import"TXHandleCrash.h"

@implementationNSDictionary (TXHandleCrash)

+(void)handleCrash

{

[TXHandleCrashhandleClass:[selfclass]exchangeClassMethod:@selector(dictionaryWithObjects:forKeys:count:)Method:@selector(handleCrashDictionaryWithObjects:forKeys:count:)];

}

+(instancetype)handleCrashDictionaryWithObjects:(constid_Nonnull__unsafe_unretained*)objects forKeys:(constid_Nonnull__unsafe_unretained*)keys count:(NSUInteger)cnt

{

idinstance =nil;

@try{

instance = [selfhandleCrashDictionaryWithObjects:objectsforKeys:keyscount:cnt];

}@catch(NSException *exception) {

[TXHandleCrashhandleException:exceptionremark:@""];

NSUIntegerindex =0;

id_Nonnull__unsafe_unretainednewObjects[cnt];

id_Nonnull__unsafe_unretainednewKeys[cnt];

for(inti =0; i

if(keys[i] && objects[i]) {

newObjects[index] = objects[i];

newKeys[index] = keys[i];

index ++ ;

}

}

instance = [selfhandleCrashDictionaryWithObjects:newObjectsforKeys:newKeyscount:index];

}@finally{

returninstance ;

}

}

@end

3.在Appdelegate中

- (BOOL)application:(UIApplication*)application didFinishLaunchingWithOptions:(NSDictionary*)launchOptions {

[TXHandleCrashstartHandle];

[[NSNotificationCenterdefaultCenter]addObserver:selfselector:@selector(handleException:)name:HandleCrashLogNotificationobject:nil];

returnYES;

}

-(void)handleException:(NSNotification*)notif

{

NSDictionary*exceptionInfo = notif.userInfo;

NSLog(@"----------%@",exceptionInfo);

}

4.在viewcontroller中

- (void)viewDidLoad {

[superviewDidLoad];

NSString*nilStr =nil;

//通过这种方法创建字典其实是调用dictionaryWithObjects:forKeys:count:方法,如果不做任何处理下面的代码就会直接崩溃,现在经常上面那么多的处理就不会崩溃了,并且可以通过AppDelegate中的通知做到收集崩溃日志的目的

NSDictionary*dictionary =@{@"key1":nilStr,@"key2":@"values"};

NSLog(@"dic=%@",dictionary);

}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,170评论 0 7
  • 本篇文章在《iOS开发之Runtime常用示例总结》基础上修改,地址是「:」http://www.cocoachi...
    小__小阅读 1,823评论 1 3
  • 这几天二宝睡得不太好,白天我没有陪他睡,在看书复习,中午饭点时也会偷懒看下电视,是爸的平板放的热播剧。 8点左右,...
    小丫屠阅读 203评论 0 1
  • 杭州是一座令人流连忘返的城市,她的美在于山、水、人情及整个城市的温柔,欣赏完美丽的景色,别忘了在双脚微酸的时候,找...
    余元明阅读 1,278评论 0 0