9.6 使用@try处理异常

好的编程实践是指能预测程序中可能出现的问题。为此,你可以测试使程序异常终止的条件并处理这些情况,可能要记录一条消息并完全终止程序,或者采取其他正确措施。例如,本章前面讲过如何测试来查看一个对象是否响应特定的消息。以避免错误为例,在程序运行时执行测试可以避免向对象发送未识别的消息。当试图发送这类未识别消息时,程序通常会立即终止并抛出一个异常。

看一下代码清单9-4。Fraction类中未定义任何名为noSuchMethod的方法。当你编译程序时,就会得到相关警告消息。

代码清单9-4


import“Fraction.h”

int main(int argc, char*argv[])

{

NSAutoreleasePool*pool=[[NSAutoreleasePool alloc]init];

Fraction*f=[[Fraction alloc]init];

[f noSuchMethod];

NSLog(@“Execution continues!”);

[f release];

[pool drain];

return 0;

}


你可以不管警告消息继续运行程序。如果这样做,程序可能会异常终止,并出现类似如下的错误:

代码清单9-4输出


-[Fraction noSuchMethod]:unrecognized selector sent to instance 0x103280

*Terminating app due to uncaught exception‘NSInvalidArgumentException’,

reason:*-[Fraction noSuchMethod]:unrecognized selector sent

to instance 0x103280

Stack:(

2482717003,

2498756859,

2482746186,

2482739532,

2482739730

Trace/BPT trap


为了避免在这类情况下程序异常终止,可以在一个特殊的语句块中加入一条或多条语句,格式如下:


@try{

statement

statement

……

}

@catch(NSException*exception){

statement

statement

……

}


在@try块中加入这些statement后,程序正常执行。但是,如果块中的某一条语句抛出异常,执行不会终止,而是立即跳到@catch块,在那里继续执行。在@catch块内可以处理异常。这里可行的执行顺序是记录出错消息、清除和终止执行。

代码清单9-5演示了异常处理。紧跟着的是程序的输出。

代码清单9-5异常处理


import“Fraction.h”

int main(int argc, char*argv[])

{

NSAutoreleasePool*pool=[[NSAutoreleasePool alloc]init];

Fraction*f=[[Fraction alloc]init];

@try{

[f noSuchMethod];

}

@catch(NSException*exception){

NSLog(@“Caught%@%@”,[exception name],[exception reason]);

}

NSLog(@“Execution continues!”);

[f release];

[pool drain];

return 0;

}


代码清单9-5输出


*-[Fraction noSuchMethod]:unrecognized selector sent to instance 0x103280

Caught NSInvalidArgumentException:*-[Fraction noSuchMethod]:

unrecognized selector sent to instance 0x103280

Execution continues!


出现异常时,@catch块被执行。包含异常信息的NSException对象作为参数传递给这个块。如你所见,name方法检索异常的名称,reason方法给出原因(运行时系统还会将原因自动输出)。

这是一个非常简单的例子,演示了如何在程序中捕获异常。可以使用@finally块包含是否执行抛出异常的@try块中的语句代码。

@throw指令允许你抛出自己的异常。可以使用该指令抛出特定的异常,或者在@catch块内抛出带你进入类似如下代码块的异常:


@throw;


自行处理异常后(例如,可能是在执行清除工作后)可能需要这么做。然后便可以让系统处理其余的工作。最后,可以使用多个@catch块按顺序捕获并处理各种异常。