可能在iOS中不太经常用到,但是了解一下还是没坏处
<span style="color: rgb(51, 51, 255);"><span style="font-size:14px;">@try{</span></span>
<span style="font-size:14px;">代码块1(可能出现异常的语句)</span>
<span style="font-size:14px;">//执行的代码,其中可能有异常。一旦发现异常,则立即跳到catch执行。否则不会执行catch里面的内容 <span style="color: rgb(51, 51, 255);"> } @catch(Exception e){</span></span>
<span style="font-size:14px;"> 代码块2(发生异常时进行处理)</span>
<span style="font-size:14px;">//除非try里面执行代码发生了异常,否则这里的代码不会执行 <span style="color: rgb(51, 51, 255);">}@finally{</span></span>
<span style="font-size:14px;"> 代码块3(始终要进行处理的语句)</span>
<span style="font-size:14px;">//不管什么情况都会执行,包括try catch 里面用了return ,可以理解为只要执行了try或者catch,就一定会执行 finally <span style="color: rgb(51, 51, 255);">}</span> catch是抓取代码块1中的异常 代码块2是出异常后的处理 代码块3是不管出不出异常都会执行,如果代1或代2中有return,代3会在return后执行</span>
<span style="font-size:14px;">总结:</span>
<span style="font-size:14px;">1、不管有木有出现异常,finally块中代码都会执行; 2、当try和catch中有return时,finally仍然会执行; 3、finally是在return后面的表达式运算后执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的; 4、finally中最好不要包含return,否则程序会提前退出,返回值不是try或catch中保存的返回值。 </span>
<span style="font-size:14px;">这个异常捕获机制就是哪里用到就写到那里</span>
以下程序已测试并通过:
设备:iOS 8模拟器中
开发工具:XCode6.1
使用@try、catch捕获异常:
以下是最简单的代码写法,其中@finally可以去掉:
1
2
3
4
5
6
7
8
9
|
@ try { // 可能会出现崩溃的代码 } @ catch (NSException *exception) { // 捕获到的异常exception } @finally { // 结果处理 } |
在这里举多一具比较详细的方法,抛出异常:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
@ try { // 1 [self tryTwo]; } @ catch (NSException *exception) { // 2 NSLog(@ "%s\n%@" , __FUNCTION__, exception); // @throw exception; // 这里不能再抛异常 } @finally { // 3 NSLog(@ "我一定会执行" ); } // 4 // 这里一定会执行 NSLog(@ "try" ); |
tryTwo方法代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
- (void)tryTwo { @ try { // 5 NSString *str = @ "abc" ; [str substringFromIndex:111]; // 程序到这里会崩 } @ catch (NSException *exception) { // 6 // @throw exception; // 抛出异常,即由上一级处理 // 7 NSLog(@ "%s\n%@" , __FUNCTION__, exception); } @finally { // 8 NSLog(@ "tryTwo - 我一定会执行" ); } // 9 // 如果抛出异常,那么这段代码则不会执行 NSLog(@ "如果这里抛出异常,那么这段代码则不会执行" ); } |
为了方便大家理解,我在这里再说明一下情况:
如果6抛出异常,那么执行顺序为:1->5->6->8->3->4
如果6没抛出异常,那么执行顺序为:1->5->7->8->9->3->4
2)部分情况的崩溃我们是无法避免的,就算是QQ也会有崩溃的时候。因此我们可以在程序崩溃之前做一些“动作”(收集错误信息),以下例子是把捕获到的异常发送至开发者的邮箱。
AppDelegate.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions: (NSDictionary *)launchOptions { // Override point for customization after application launch. NSSetUncaughtExceptionHandler(&UncaughtExceptionHandler); return YES; } void UncaughtExceptionHandler(NSException *exception) { /** * 获取异常崩溃信息 */ NSArray *callStack = [exception callStackSymbols]; NSString *reason = [exception reason]; NSString *name = [exception name]; NSString *content = [NSString stringWithFormat:@ "========异常错误报告========\nname:%@\nreason: \n%@\ncallStackSymbols:\n%@" ,name,reason,[callStack componentsJoinedByString:@ "\n" ]]; /** * 把异常崩溃信息发送至开发者邮件 */ NSMutableString *mailUrl = [NSMutableString string]; [mailUrl appendString:@ "mailto:test@qq.com" ]; [mailUrl appendString:@ "?subject=程序异常崩溃,请配合发送异常报告,谢谢合作!" ]; [mailUrl appendFormat:@ "&body=%@" , content]; // 打开地址 NSString *mailPath = [mailUrl stringByAddingPercentEscapesUsingEncoding:NSUTF8StringEncoding]; [[UIApplication sharedApplication] openURL:[NSURL URLWithString:mailPath]]; } |