IOS疯狂基础之 Block

来源:互联网 发布:更改windows桌面路径 编辑:程序博客网 时间:2024/04/19 21:45

基础扫盲如下解释block:

Block是iOS4.0+ 和Mac OS X 10.6+ 引进的对C语言的扩展,用来实现匿名函数的特性。

用维基百科的话来说,Block是Apple Inc.为C、C++以及Objective-C添加的特性,使得这些语言可以用类lambda表达式的语法来创建闭包

用Apple文档的话来说,A block is an anonymous inline collection of code, and sometimes also called a "closure".

关于闭包,我觉得阮一峰的一句话解释简洁明了:闭包就是能够读取其它函数内部变量的函数

这个解释用到block来也很恰当:一个函数里定义了个block,这个block可以访问该函数的内部变量。


Block除了能够定义参数列表、返回类型外,还能够获取被定义时的词法范围内的状态(比如局部变量),并且在一定条件下(比如使用__block变量)能够修改这些状态。此外,这些可修改的状态在相同词法范围内的多个block之间是共享的,即便出了该词法范围(比如栈展开,出了作用域),仍可以继续共享或者修改这些状态。

通常来说,block都是一些简短代码片段的封装,适用作工作单元,通常用来做并发任务、遍历、以及回调。


而在很多框架中,block越来越经常被用作回调函数,取代传统的回调方式。

  • 用block作为回调函数,可以使得程序员在写代码更顺畅,不用中途跑到另一个地方写一个回调函数,有时还要考虑这个回调函数放在哪里比较合适。采用block,可以在调用函数时直接写后续处理代码,将其作为参数传递过去,供其任务执行结束时回调。
  • 另一个好处,就是采用block作为回调,可以直接访问局部变量。比如我要在一批用户中修改一个用户的name,修改完成后通过回调更新对应用户的单元格UI。这时候我需要知道对应用户单元格的index,如果采用传统回调方式,要嘛需要将index带过去,回调时再回传过来;要嘛通过外部作用域记录当前操作单元格的index(这限制了一次只能修改一个用户的name);要嘛遍历找到对应用户。而使用block,则可以直接访问单元格的index。

这份文档中提到block的几种适用场合:

  • 任务完成时回调处理
  • 消息监听回调处理
  • 错误回调处理
  • 枚举回调
  • 视图动画、变换
  • 排序

声明和使用Block 
Apple文档中介绍了如何将一个Block声明为变量,并将其作为一个函数使用:

  1. int (^oneFrom)(int) = ^(int anInt) {  
  2.     return anInt - 1;  
  3. };  
  4. // 我们创建了一个内联块^(int anInt)... ,其函数体和结果被传到了另外一个名为OneFrom的Block。  
  5.  
  6. printf("1 from 10 is %d", oneFrom(10));  
  7. // 打印出: "1 from 10 is 9"  
  8. // 这个block函数(distanceTraveled)传入3个float型参数,返回float值。   
  9.  
  10. float (^distanceTraveled) (float, float, float) =  
  11.  
  12.                           ^(float startingSpeed, float acceleration, float time) {  
  13.     float distance = (startingSpeed * time) + (0.5 * acceleration * time * time);  
  14.     return distance;  
  15. }; 


  你也可以传入一个Block作为一个参数,而不要以如上的方式声明它们,这样就可以在需要将block作为参数的时候以内联代码的方式简单地实现。

  1. NSArray *anArray = [NSArray arrayWithObjects: @"cat", @"dog",nil];  
  2. sortFunction(anArray, ^(string *a string *b){  
  3. if ( a == @"cat"return TRUE; }); 


这样我们就看到一个内联的block代码段占据了最后一个参数(必须是参数列表的最后一个参数)的位置。Cocoa提供了很多使用Block的方法,这样你就可以传入Block作为方法的参数:

  1. NSArray *array = [NSArray arrayWithObjects: @"A", @"B", @"C",  nil];  
  2. NSSet *filterSet = [NSSet setWithObjects: @"A", @"Z", @"Q", nil];  
  3.  
  4. BOOL (^test)(id obj, NSUInteger idx, BOOL *stop); //Block declaration returns BOOL, params inc. id and BOOL  
  5. //body of block gets the block literal ^(id obj, NSUInteger idx, Bool *stop)... and the body logic   
  6. test = ^ (id obj, NSUInteger idx, BOOL *stop) {  
  7.     if (idx < 5) {  
  8.         if ([filterSet containsObject: obj]) {  
  9.             return YES;  
  10.         }  
  11.     }  
  12.     return NO;  
  13.  
  14. }; 

Apple提供的另外一个例子是:

  1. __block BOOL found = NO;  
  2. NSSet *aSet = [NSSet setWithObjects: @"Alpha", @"Beta", @"Gamma", @"X", nil];  
  3. NSString *string = @"gamma";  
  4. //we provide below a way of how to enumerate, using our own compare logic  
  5. [aSet enumerateObjectsUsingBlock:^(id obj, BOOL *stop) {  
  6.     if ([obj localizedCaseInsensitiveCompare:string] == NSOrderedSame) {  
  7.         *stop = YES;  
  8.         found = YES;  
  9.     }  
  10. }];