关于阅读_关于Objective-C错误和异常处理的经验总结

更新时间:2020-01-17    来源:班主任工作总结    手机版     字体:

【www.bbyears.com--班主任工作总结】

在 Objective-C 开发中,异常往往是由程序员的错误导致的 app 无法继续运行,比如我们向一个无法响应某个消息的 NSObject 对象发送了这个消息,会得到 NSInvalidArgumentException 的异常,并告诉我们 "unrecognized selector sent to instance";比如我们使用一个超过数组元素数量的下标来试图访问 NSArray 的元素时,会得到 NSRangeException。类似由于这样所导致的程序无法运行的问题应该在开发阶段就被全部解决,而不应当出现在实际的产品中。相对来说,由 NSError 代表的错误更多地是指那些“合理的”,在用户使用 app 中可能遇到的情况:比如登陆时用户名密码验证不匹配,或者试图从某个文件中读取数据生成 NSData 对象时发生了问题 (比如文件被意外修改了) 等等。

但是 NSError 的使用方式其实变相在鼓励开发者忽略错误。想一想在使用一个带有错误指针的 API 时我们做的事情吧。我们会在 API 调用中产生和传递 NSError,并藉此判断调用是否失败。作为某个可能产生错误的方法的使用者,我们用传入 NSErrorPointer 指针的方式来存储错误信息,然后在调用完毕后去读取内容,并确认是否发生了错误。比如在 Objective-C 中,我们会写类似这样的代码:

NSError *error;  
BOOL success = [data writeToFile: path options: options error: &error];  
if(error) {  
    // 发生了错误
}

这非常棒,但是有一个问题:在绝大多数情况下,这个方法并不会发生什么错误,而很多工程师也为了省事和简单,会将输入的 error 设为 nil,也就是不关心错误 (因为可能他们从没见过这个 API 返回错误,也不知要如何处理)。于是调用就变成了这样:

[data writeToFile: path options: options error: nil];


但是事实上这个 API 调用是会出错的,比如设备的磁盘空间满了的时候,写入将会失败。但是当这个错误出现并让你的 app 陷入难堪境地的时候,你几乎无从下手进行调试 -- 因为系统曾经尝试过通知你出现了错误,但是你却选择视而不见。

在 Swift 2.0 中,Apple 为这么语言引入了异常机制。现在,这类带有 NSError 指针作为参数的 API 都被改为了可以抛出异常的形式。比如上面的 writeToFile:options:error:,在 Swift 中变成了:

public func writeToFile(path: String, options writeOptionsMask: NSDataWritingOptions) throws


我们在使用这个 API 的时候,不再像之前那样传入一个 error 指针去等待方法填充,而是变为使用 try catch 语句:

do {  
    try d.writeToFile("Hello", options: [])
} catch let error as NSError {
    print ("Error: \\(error.domain)")
}


如果你不使用 try 的话,是无法调用 writeToFile: 方法的,它会产生一个编译错误,这让我们无法有意无意地忽视掉这些错误。在上面的示例中 catch 将抛出的异常 (这里就是个 NSError) 用 let 进行了类型转换,这其实主要是针对 Cocoa 现有的 API 的,是对历史的一种妥协。对于我们新写的可抛出异常的 API,我们应当抛出一个实现了 ErrorType 的类型,enum 就非常合适,举个例子:

enum LoginError: ErrorType {  
    case UserNotFound, UserPasswordNotMatch
}
func login(user: String, password: String) throws {  
    //users 是 [String: String],存储[用户名:密码]
    if !users.keys.contains(user) {
        throw LoginError.UserNotFound
    }
    if users[user] != password {
        throw LoginError.UserPasswordNotMatch
    }
    print("Login successfully.")
}


这样的 ErrorType 可以非常明确地指出问题所在。在调用时,catch 语句实质上是在进行模式匹配:

do {  
    try login("onevcat", password: "123")
} catch LoginError.UserNotFound {
    print("UserNotFound")
} catch LoginError.UserPasswordNotMatch {
    print("UserPasswordNotMatch")
}
// Do something with login user


如果你之前写过 Java 或者 C# 的话,会发现 Swift 中的 try catch 块和它们中的有些不同。在那些语言里,我们会把可能抛出异常的代码都放在一个 try 里,而 Swift 中则是将它们放在 do 中,并只在可能发生异常的语句前添加 try。相比于 Java 或者 C# 的方式,Swift 里我们可以更清楚地知道是哪一个调用可能抛出异常,而不必逐句查阅文档。

当然,Swift 现在的异常机制也并不是十全十美的。最大的问题是类型安全,不借助于文档的话,我们现在是无法从代码中直接得知所抛出的异常的类型的。比如上面的 login 方法,光看方法定义我们并不知道 LoginError 会被抛出。一个理想中的异常 API 可能应该是这样的:

func login(user: String, password: String) throws LoginError


很大程度上,这是由于要与以前的 NSError 兼容所导致的妥协,对于之前的使用 NSError 来表达错误的 API,我们所得到的错误对象本身就是用像 domain 或者 error number 这样的属性来进行区分和定义的,这与 Swift 2.0 中的异常机制所抛出的直接使用类型来描述错误的思想暂时是无法兼容的。不过有理由相信随着 Swift 的迭代更新,这个问题会在不久的将来得到解决。

另一个限制是对于非同步的 API 来说,抛出异常是不可用的 -- 异常只是一个同步方法专用的处理机制。Cocoa 框架里对于异步 API 出错时,保留了原来的 NSError 机制,比如很常用的 NSURLSession 中的 dataTask API:

func dataTaskWithURL(_ url: NSURL,  
   completionHandler completionHandler: ((NSData!,
                              NSURLResponse!,
                              NSError!) -> Void)?) -> NSURLSessionDataTask


对于异步 API,虽然不能使用异常机制,但是因为这类 API 一般涉及到网络或者耗时操作,它所产生错误的可能性要高得多,所以开发者们其实无法忽视这样的错误。但是像上面这样的 API 其实我们在日常开发中往往并不会去直接使用,而会选择进行一些封装,以求更方便地调用和维护。一种现在比较常用的方式就是借助于 enum。作为 Swift 的一个重要特性,枚举 (enum) 类型现在是可以与其他的实例进行绑定的,我们还可以让方法返回枚举类型,然后在枚举中定义成功和错误的状态,并分别将合适的对象与枚举值进行关联:

enum Result {  
    case Success(String)
    case Error(NSError)
}
func doSomethingParam(param:AnyObject) -> Result {  
    //...做某些操作,成功结果放在 success 中
    if success {
        return Result.Success("成功完成")
    } else {
        let error = NSError(domain: "errorDomain", code: 1, userInfo: nil)
        return Result.Error(error)
    }
}


在使用时,利用 switch 中的 let 来从枚举值中将结果取出即可:

let result = doSomethingParam(path)
switch result {  
    case let .Success(ok):
    let serverResponse = ok
case let .Error(error):  
    let serverResponse = error.description
}


在 Swift 2.0 中,我们甚至可以在 enum 中指定泛型,这样就使结果统一化了。

enum Result {  
    case Success(T)
    case Failure(NSError)
}


我们只需要在返回结果时指明 T 的类型,就可以使用同样的 Result 枚举来代表不同的返回结果了。这么做可以减少代码复杂度和可能的状态,同时不是优雅地解决了类型安全的问题,可谓一举两得。

因此,在 Swift 2 时代中的错误处理,现在一般的最佳实践是对于同步 API 使用异常机制,对于异步 API 使用泛型枚举。



Objective-C语法之异常处理

异常处理捕获的语法

@try {
    <#statements#>
}
@catch (NSException *exception) {
    <#handler#>
}
@finally {
    <#statements#>
}


@catch{} 块 对异常的捕获应该先细后粗,即是说先捕获特定的异常,再使用一些泛些的异常类型。

我们自定义两个异常类,看看异常异常处理的使用。

1、新建SomethingException,SomeOverException这两个类,都继承与NSException类。

SomethingException.h
#import 
@interface SomethingException : NSException
@end
SomethingException.m
#import "SomethingException.h"
@implementation SomethingException
@end
SomeOverException.h
#import 
@interface SomeOverException : NSException
@end
SomeOverException.m
#import "SomeOverException.h"
@implementation SomeOverException
@end


2、新建Box类,在某些条件下产生异常。

#import 
@interface Box : NSObject
{
    NSInteger number;
}
-(void) setNumber: (NSInteger) num;
-(void) pushIn;
-(void) pullOut;
-(void) printNumber;
@end
@implementation Box
-(id) init {
    self = [super init];
    
    if ( self ) {
        [self setNumber: 0];
    }
    
    return self;
}
-(void) setNumber: (NSInteger) num {
    number = num;
    
    if ( number > 10 ) {
        NSException *e = [SomeOverException
                          exceptionWithName: @"BoxOverflowException"
                          reason: @"The level is above 100"
                          userInfo: nil];
        @throw e;
    } else if ( number >= 6 ) {
        // throw warning
        NSException *e = [SomethingException
                          exceptionWithName: @"BoxWarningException"
                          reason: @"The level is above or at 60"
                          userInfo: nil];
        @throw e;
    } else if ( number < 0 ) {
        // throw exception
        NSException *e = [NSException
                          exceptionWithName: @"BoxUnderflowException"
                          reason: @"The level is below 0"
                          userInfo: nil];
        @throw e;
    }
}
-(void) pushIn {
    [self setNumber: number + 1];
}
-(void) pullOut {
    [self setNumber: number - 1];
}
-(void) printNumber {
    NSLog(@"Box number is: %d", number);
}
@end


这个类的作用是,初始化Box时,number数字是0,可以用pushIn 方法往Box里推入数字,每调用一次,number加1.当number数字大于等于6时产生SomethingException异常,告诉你数字达到或超过6了,超过10时产生SomeOverException异常,小于1时产生普通的NSException异常。

这里写 [SomeOverException  exceptionWithName:@"BoxOverflowException"  reason:@"The level is above 100"异常的名称和理由,在捕获时可以获取。

3、使用Box,在适当添加下捕获Box类的异常

3.1、在没超过6时,没有异常

- (void)viewDidLoad
{
    [super viewDidLoad];    
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    Box *box = [[Box alloc]init];
    for (int i = 0; i < 5; i++) {
        [box pushIn];
        [box printNumber];
    }
}


打印结果:

Box number is: 1

Box number is: 2

Box number is: 3

Box number is: 4

Box number is: 5

3.2 超过6,产生异常


for (int i = 0; i < 11; i++) {
        [box pushIn];
        [box printNumber];
}


2012-07-04 09:12:05.889 ObjectiveCTest[648:f803] Box number is: 1
2012-07-04 09:12:05.890 ObjectiveCTest[648:f803] Box number is: 2
2012-07-04 09:12:05.890 ObjectiveCTest[648:f803] Box number is: 3
2012-07-04 09:12:05.890 ObjectiveCTest[648:f803] Box number is: 4
2012-07-04 09:12:05.891 ObjectiveCTest[648:f803] Box number is: 5
2012-07-04 09:12:05.891 ObjectiveCTest[648:f803] *** Terminating app due to uncaught exception 'BoxWarningException', reason: 'The number is above or at 60'


这是时,程序抛出异常崩溃了。那怎么使程序不崩溃呢,做异常处理。

3.3、加上异常处理

for (int i = 0; i < 11; i++) {
    @try {
        [box pushIn];
    }
    @catch (SomethingException *exception) {
        NSLog(@"%@ %@", [exception name], [exception reason]);
    }
    @catch (SomeOverException *exception) {
        NSLog(@"%@", [exception name]);
    }
    @finally {
        [box printNumber];
    }
}


运行,程序没有崩溃,打印结果:

2012-07-04 09:14:35.165 ObjectiveCTest[688:f803] Box number is: 1
2012-07-04 09:14:35.167 ObjectiveCTest[688:f803] Box number is: 2
2012-07-04 09:14:35.167 ObjectiveCTest[688:f803] Box number is: 3
2012-07-04 09:14:35.167 ObjectiveCTest[688:f803] Box number is: 4
2012-07-04 09:14:35.167 ObjectiveCTest[688:f803] Box number is: 5
2012-07-04 09:14:35.167 ObjectiveCTest[688:f803] BoxWarningException The number is above or at 60
2012-07-04 09:14:35.168 ObjectiveCTest[688:f803] Box number is: 6
2012-07-04 09:14:35.168 ObjectiveCTest[688:f803] BoxWarningException The number is above or at 60
2012-07-04 09:14:35.168 ObjectiveCTest[688:f803] Box number is: 7
2012-07-04 09:14:35.168 ObjectiveCTest[688:f803] BoxWarningException The number is above or at 60
2012-07-04 09:14:35.168 ObjectiveCTest[688:f803] Box number is: 8
2012-07-04 09:14:35.168 ObjectiveCTest[688:f803] BoxWarningException The number is above or at 60
2012-07-04 09:14:35.169 ObjectiveCTest[688:f803] Box number is: 9
2012-07-04 09:14:35.169 ObjectiveCTest[688:f803] BoxWarningException The number is above or at 60
2012-07-04 09:14:35.169 ObjectiveCTest[688:f803] Box number is: 10
2012-07-04 09:14:35.169 ObjectiveCTest[688:f803] BoxOverflowException
2012-07-04 09:14:35.225 ObjectiveCTest[688:f803] Box number is: 11


超过10时,SomeOverException异常抛出。

3.4 、小于0时的异常


在Box类的setNumber里,当number小于0时,我们抛出普通异常。

@try {
    [box setNumber:-10];
}
@catch (NSException *exception) {
    NSLog(@"%@",[exception name]);
}
@finally {
    [box printNumber];
}



打印结果

2012-07-04 09:17:42.405 ObjectiveCTest[753:f803] BoxUnderflowException
2012-07-04 09:17:42.406 ObjectiveCTest[753:f803] Box number is: -10

本文来源:http://www.bbyears.com/banzhurengongzuo/84614.html

热门标签

更多>>

本类排行