
defer 语句在 Go 语言中是一个非常方便的特性,它允许开发者将清理操作(如关闭文件、释放资源等)放在函数开头,而无需担心在函数的不同返回点忘记执行这些操作。Objective-C 本身并没有直接对应的语法,但我们可以利用 Objective-C 的特性来模拟实现类似的功能。
基于 @finally 块的实现
Objective-C 的 @try / @catch / @finally 结构提供了一种在代码块执行完毕后(无论是否发生异常)执行特定代码的方式。我们可以利用 @finally 块来实现 defer 语句的效果。
以下是一种基于 @finally 块和宏定义的实现方案:
#define SCOPE {id _defered_actions__=[[NSMutableArray alloc]init];@try{
#define END_SCOPE }@finally{for(void(^action)()in[_defered_actions__ reverseObjectEnumerator])action();[_defered_actions__ release];}}
#define DEFER_COPY(_code__) {id _blk__=[^{_code__;}copy];[_defered_actions__ addObject:_blk__];[_blk__ release];}
#define DEFER(_code__) ([_defered_actions__ addObject:(^{_code__;})])代码解释:
- SCOPE:定义了一个名为 _defered_actions__ 的可变数组,用于存储需要延迟执行的代码块。同时,开启一个 @try 块。
- END_SCOPE:结束 @try 块,并开启 @finally 块。在 @finally 块中,遍历 _defered_actions__ 数组,并逆序执行其中存储的所有代码块。最后,释放 _defered_actions__ 数组。
- DEFER(_code__):将一个代码块 _code__ 添加到 _defered_actions__ 数组中。这个代码块将在函数返回前执行。
- DEFER_COPY(_code__):与 DEFER 类似,但它会先拷贝代码块 _code__,然后再添加到 _defered_actions__ 数组中。这在需要捕获局部变量的值时非常有用。
使用示例:
本文档主要讲述的是Android中JNI编程的那些事儿;JNI译为Java本地接口。它允许Java代码和其他语言编写的代码进行交互。在android中提供JNI的方式,让Java程序可以调用C语言程序。android中很多Java类都具有native接口,这些接口由本地实现,然后注册到系统中。希望本文档会给有需要的朋友带来帮助;感兴趣的朋友可以过来看看
@interface XXObject : NSObject
-(int)factorial:(int)x;
@end
@implementation XXObject
-(int)factorial:(int)x { SCOPE
printf("begin foo:%d\n", x);
DEFER( printf("end foo:%d\n", x) );
if (x > 0)
return x * [self factorial:x-1];
else if (x == 0)
return 1;
else {
@throw [NSException exceptionWithName:@"NegativeFactorialException"
reason:@"Cannot call factorial on negative numbers"
userInfo:nil];
return 0;
}
END_SCOPE }
-(void)dealloc {
printf("%p has been released.\n", self);
[super dealloc];
}
@end
void do_stuff() { SCOPE
__block XXObject* x = [[XXObject alloc] init];
DEFER({
printf("releasing %p.\n", x);
[x release];
});
int i;
for (i = 2; i >= -1; -- i) {
// use DEFER_COPY to retain the local variable 'i' and 'fact'
int fact = [x factorial:i];
DEFER_COPY( printf("%d! == %d\n", i, fact) );
}
END_SCOPE }
int main () {
NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init];
@try {
do_stuff();
} @catch(NSException* e) {
// note that the @finally statements might not be called in 64-bit if we
// left the exception uncaught.
NSLog(@"%@", e);
}
[pool drain];
return 0;
}注意事项:
- 必须成对使用 SCOPE 和 END_SCOPE。
- DEFER_COPY 适用于需要捕获局部变量值的情况,例如在循环中使用 DEFER 时。
- 由于该实现依赖于 @finally 块,因此在某些情况下(例如未捕获的异常),@finally 块可能不会被执行。在64位架构中,如果异常未被捕获,@finally 语句可能不会被调用。
- 使用宏会稍微降低代码的可读性,请谨慎使用。
总结:
通过使用 Objective-C 的 @finally 块和宏定义,我们可以实现类似于 Go 语言中 defer 语句的功能。这种方法可以在一定程度上简化代码,并确保在函数返回前执行必要的清理操作。但是,需要注意 @finally 块的执行条件以及宏定义对代码可读性的影响。在实际应用中,请根据具体情况权衡利弊,选择最适合的方案。









