为什么我选择使用 Blocks(块)

时间:2022-11-12 18:21:12

为什么我选择使用 Blocks(块) 



来源:shawenlx

链接:http://www.jianshu.com/p/a4dd57e32937


  • 扯淡:到了新公司接手新框架之后,发现大量的使用Blocks,之前很多时候都是使用代理,突然面对这个陌生的语法,特地科普总结了一番。




什么是Blocks


  • 一句话概括就是,带有局部变量的匿名函数(即不带名称的函数)。也称为闭包。


Blocks语法


  • ^ 返回值类型 参数列表 表达式


//例如

^int(intcount){returncount + 1;}


上述表达式中,返回值类型 以及 参数列表可以省略(如下), 语法加粗表明不可缺少的部分。


^{

    printf("Hello, blocks");

}




将Blocks声明为属性


  • 多数情况下,我们会把Blocks声明为属性,便于调用。可以利用关键字typedef 对Blocks进行别名。


typedef返回值类型(^Blocks别名)(参数列表);

typedefint(^blk_t)(int);


//将Blocks声明为属性

@property(nonatomic,copy)blk_tblk;


//对Blocks进行声明定义

blk = ^(intcount){returncount + 1;};


//调用Blocks 和 调用C语言声明函数方式一样

NSLog(@"%d\n",blk(10));


将Blocks作为参数传入方法中


  • 通常都会事先为Blocks取一个别名,一来方便理解,好吧不装逼,就是偷懒。


//定义:

typedefvoid(^Block)(inta);

-(void)nslogParameterWithBlock:(Block)block{

    if(block){

        block(10);

    }

}


//赋值并调用

[object nslogParameterWithBlock:^(inta){

    NSLog(@"%d",a);

}];




Blocks截获自动变量值


  • 什么是截获自动变量值,所谓自动变量,就是局部变量。那为什么会截获呢?先来看个例子。


intval = 10;

void  (^block)(void) = ^ {printf("val = %d",val);};

val = 20;

block();  //此处调用block,打印为val = 10;


  • 之所以没有打印val = 20,是因为该Blocks在编译期就进行了初始化,对声明的val进行截获,也就是说,把val拷贝一份存入Blocks中。所以当外部val进行了修改时,不影响Blocks截获的val值。




__block 关键字


  • 从上个例子中,我们知道Blocks会对变量进行截获。那如果我非要在Blocks中修改外部参数怎么办呢?我们可以通过__block关键字做到!


__blockintval = 10;

void  (^block)(void) = ^ {printf("val = %d",val);};

val = 20;

block();  //此处调用block,打印为val = 20;


  • 通过__block说明符对外部参数进行声明后,实际上,相当于在声明定义Blocks的时候,不是通过拷贝内存的方式拷贝外部参数,而是通过指针指向该外部变量,从而达到可以在Blocks中修改的效果。




总结:


  • Blocks使用方便,但是block的内存管理确实是一项非常重要也非常容易出错的地方,稍不注意便会造成内存泄露,所以,在使用Blocks的时候,也要小心。

  • 因为是匿名函数,所以在项目中多人合作的时候,最后写上每个Blocks的注释说明。

  • 后续将继续总结Blocks的内存管理内容。