Block

基本使用

<returntype> (^blockname) (list of arguments) = ^(arguments) {body ;} ;

其中返回类型和参数可以省略

声明

按照<returnValue> (^blockName) (parameters)的方式进行block声明未免麻烦了些,可以通过关键字typedef为block类型命名,然后直接通过类型名进行block创建。

  • 命名
typedef <returntype> (^blockname)(list of arguments) ;
  • 创建
blockname block = ^(arguments) {body ;} ;

__block关键字

CGPoint center = CGPointZero ;
CGPoint (^pointAddHandler)(CGPoint addPoint) = ^(CGPoint addPoint) {
     return CGPointMake(center.x + addPoint.x, center.y + addPoint.y) ;
 }
center = CGPointMake(100, 100) ;
NSLog(@"%@", pointAddHandler(CGPointMake(10, 10))) ;   //输出{10,10}

block在捕获变量的时候会保存变量被捕获时的状态(对象变量除外),之后即便变量再次改变,block中变量的值也不会发生改变。所以上述代码在计算新的坐标值时center的值依旧为CGPointZero。如果希望在block中修改外界的本地变量,可以通过给这些变量加上__block关键字来实现。

循环引用

如果A创建并引用了B,B引用了callBackBlock,而callBackBlock中又引用了A,那么就会形成循环引用。解决方法是使用弱引用来解除这个循环:__weak typeof(A) weakA = A ;。但是如何理解block引起的循环引用问题呢?

创建一个BlockTestObject类,它的两个属性如下:

@property (nonatomic, copy) NSString *str ;
@property (nonatomic, copy) void(^myBlock)() ;

测试代码如下:

BlockTestObject *myTest = [[BlockTestObject alloc] init] ;
myTest.str = @"这是一个测试";
myTest.myBlock= ^{
    NSLog(@"%@",myTest.str) ;
} ;
myTest.myBlock() ;

如果block代码块的内部使用了外部的强引用对象,那么block代码块内部就会自动生成一个强引用指向该对象。上述代码中,myBlock会自动生成一个强引用指向myTest对象,而myTest对象又有强引用指向myBlock,于是便造成了循环引用,使myTest对象无法被销毁。

解决这个问题常用方法就是使用 __weak
添加宏#define weakSelf(object) __weak typeof(object) weak##object = object ;,测试代码修改为:

BlockTestObject *myTest = [[BlockTestObject alloc] init] ;
myTest.str = @"这是一个测试" ;
weakSelf(myTest) ;
myTest.myBlock = ^{
  NSLog(@"%@",weakmyTest.str) ;
} ;
myTest.myBlock() ;

如果block代码块的内部使用了外部的弱引用对象,那么block代码块内部就会自动生成一个弱引用指向该对象。上述代码中,myBlock使用了弱引用对象weakmyTest,因此myBlock只会生成一个弱引用指向对象myTest,从而不会造成循环引用。

由于仅有一个弱引用指向对象myTest,因此如果myBlock中的代码出现延时执行的情况,那么在该代码执行前对象myTest就有可能被销毁。测试代码修改为:

BlockTestObject *myTest = [[BlockTestObject alloc] init] ;
myTest.str = @"这是一个测试" ;
weakSelf(myTest) ;
    myTest.myBlock = ^{
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            NSLog(@"%@",weakmyTest.str) ;
        }) ;
    } ;
myTest.myBlock() ;

此时打印的结果为(null),对象myTest在打印前就已经销毁。可以通过 __weak__strong一起使用来解决这个问题。添加宏#define strongSelf(object) __strong typeof(object) object = weak##object ;,测试代码修改为:

BlockTestObject *myTest = [[BlockTestObject alloc] init] ;
myTest.str = @"这是一个测试" ;
weakSelf(myTest)
myTest.myBlock = ^{
  strongSelf(myTest)
  dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0*NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
    NSLog(@"%@",myTest.str) ;
  }) ;
} ;
myTest.myBlock() ;

具体分析如下:

  1. 回调执行strongSelf(myTest)这行代码。一方面,由于使用了外部的弱引用对象weakmyTest,因此会自动生成一个弱引用指向对象myTest。另一方面,block内部定义的局部变量strongSelf(myTest)会生成一个强引用指向对象myTest。
  2. GCD的dispatch_after代码块使用了该代码块外部的强引用对象myTest,因此会产生强引用指向对象myTest。
  3. dispatch_after代码块会延迟2秒执行,但是并不会阻塞线程,因此myBlock会继续执行。当myBlock执行完成时,内部的局部变量strongSelf(myTest)就会销毁,此时myBlock内部指向对象myTest的强引用也会销毁。
  4. 最后只剩下GCD的dispatch_after代码块有强引用指向对象myTest,所以对象myTest没有被销毁。当延时时间结束,dispatch_after代码块执行完成后就不会再有强引用指向对象myTest。
  5. 没有强引用指向的对象myTest就会销毁。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 前言 Blocks是C语言的扩充功能,而Apple 在OS X Snow Leopard 和 iOS 4中引入了这...
    小人不才阅读 3,790评论 0 23
  • iOS代码块Block 概述 代码块Block是苹果在iOS4开始引入的对C语言的扩展,用来实现匿名函数的特性,B...
    smile刺客阅读 2,377评论 2 26
  • 在介绍Block之前通过一个简单的应用场景认识下Block 场景描述如下:TableView上面有多个Custom...
    黑_白_灰阅读 1,427评论 4 29
  • 近来把《iOS与OS X多线程和内存管理》这本书又掏出来看了一遍,这本书前前后后加起来看了能有三四遍了,每次看都有...
    老司机Wicky阅读 2,358评论 5 46
  • 一、Objective-C发展史 Objective-C从1983年诞生,已经走过了30多年的历程。随着时间的推移...
    没事蹦蹦阅读 5,915评论 12 34