Block基本概念
本小节知识点:
- 【了解】什么是Block
- 【理解】block的格式
1.什么是Block
-
Block是iOS中一种比较特殊的数据类型
-
Block是苹果官方特别推荐使用的数据类型, 应用场景比较广泛
- 动画
- 多线程
- 集合遍历
- 网络请求回调
-
Block的作用
- 用来保存某一段代码, 可以在恰当的时间再取出来调用
- 功能类似于函数和方法
2.block的格式
- Block的定义格式
返回值类型 (^block变量名)(形参列表) = ^(形参列表) {};
- block最简单形式
void (^block名)() = ^{代码块;}例如:void (^myBlock)() = ^{ NSLog(@"李南江"); };
- block带有参数的block的定义和赋值
void (^block名称)(参数列表)= ^ (参数列表) { // 代码实现; }例如:void (^myBlock)(int) = ^(int num){ NSLog(@"num = %i", num); };
- 带有参数和返回值的block
返回类型 (^block名称)(参数列表)= ^ (参数列表) { // 代码实现; }例如:int (^myBlock)(int, int) = ^(int num1, int num2){ return num1 + num2; };
- 调用Block保存的代码
block变量名(实参);
block和typedef的使用
C语言的指向函数指针用法
#importint sum(int value1,int value2){ return value1 + value2;}int minus(int value1,int value2){ return value1 - value2;}int main(int argc,onst char * argv[]){ int (*sumP)(int,int); sumP = sum; NSLog(@"sum = %i",sumP(20,10)); int (*minusP)(int,int) minusP = minus; NSLog(@"minus = %i",minusP(20,10)); return 0;}
由于相同的部分可以抽离出来,所以需要使用typedef
#importint sum(int value1,int value2){ return value1 + value2;}int minus(int value1,int value2){ return value1 - value2;}typedef int (*calculte)(int,int)int main(int argc,onst char * argv[]){ calculte sumP = sum; NSLog(@"sum = %i",sumP(20,10)); calculte minusP = minus; NSLog(@"minus = %i",minusP(20,10)); return 0;}
OC的block的用法
#importint main(int argc,onst char * argv[]){ int (^sumBlock)(int,int); sumBlock = ^(int value1,int value2){ return value1 + value2; }; NSLog(@"sum = %i",sumBlock(20,10)); int (^minBlock)(int,int); minBlock = ^(int value1,int value2){ return value1 - value2; }; NSLog(@"min = %i",minBlock(20,10)); return 0;}
使用typedef之后:
#importtypedef int (^calculteBlock)(int,int);int main(int argc,onst char * argv[]){ calculteBlock sumBlock = ^(int value1,int value2){ return value1 + value2; }; NSLog(@"sum = %i",sumBlock(20,10)); calculteBlock minBlock = ^(int value1,int value2){ return value1 - value2; }; NSLog(@"min = %i",minBlock(20,10)); return 0;}
block的应用场景
在学习UI之后就会体会到block的好处。
这段代码要多练习,练习到不用看源码就能熟练敲出来。
#importvoid Hello(void (^func)()){ NSLog(@"Hello Wrold~"); func(); NSLog(@"Hello HeYang~");}void sayHello(){ Hello(^{ NSLog(@"Hello World,Hello HeYang~"); });}int main(int argc, const char * argv[]) { @autoreleasepool { sayHello(); } return 0;}
举个block应用实例场景:
需求: 找到需要读取并需要操作的文件步骤: 读取文件 操作文件 关闭文件分析: 其中读取文件和关闭文件都是一样的 而操作文件,会根据需求而变化,或者根据文件类型而不一样总结: 这里就需要使用block把操作文件的功能分离出来。
将Objc的m文件编译成C++文件的终端命令:
cc -rewrote-objc xx.m(OC的m文件)
然后在当前目录下就会出现后缀cpp的C++文件。
block是存储在堆中还是在栈中?默认情况下block存储在栈中,如果对block进行一个copy操作,block会转移到堆中如果block在栈中,block中访问了外界的对象,那么不会对对象进行retain操作但如果是block在堆中,block中访问了外界的对象,那么会对外界的对象进行一次retain
在MRC模式下,我们分析一下是不是会出现内存泄露
Person *p = [[Person alloc]init];NSLog(@"retainCount = %lu",[p retainCount]);void (^myBlock)() = ^{ NSLog(@"a = %@",p); NSLog(@"block retainCount = %lu",[p retain]);}Block_copy(myBlock);myBlock();
如果在block中访问了外界的对象,一定要给对象加上__block,只要加上了__block,哪怕block在堆中,也不会对外界的对象进行retain。
加了__block就是地址传递,不加__block是值传递。正因为是值传递,所以可以访问其值,或者return返回其值。