1,Block的由来
学习任何一种知识,我们必须要了解他的由来,这样才能更好的接受和掌握。
Block是iOS4.0以后引进的对C语言的扩展,用来实现匿名函数的特性。Apple文档上说:“A block is an anonymous inline collection of code, and sometimes also called a "closure”,关于闭包,曾看过一句话:闭包就是能够读取其它函数内部变量的函数。
其实Block和闭包有很多相似的地方,一个函数里定义了一个Block,这个Block就可以访问这个函数内部的变量了。
2,Block的定义
1>block是可以用来保存一段代码或者说封装一段代码,-->代码块
2>block的标志是^
3>block跟函数很像(可以有返回值,可以有参数,使用时必须调用)
定义一个Block:
返回值类型 (^名称)(参数类型...) = ^(参数类型:参数名...){
需要封装的代码块;
}
例如:定义一个有参有返回值的Block
NSInteger (^SumNum1AndNum2)(int,int) = ^(int:num1,int num2){
return num1 + num2;
}
调用Block :int sum = SumNum1AndNum2(100,200);
*如果block没有参数,那么他右边的()可以省略
*使用宏来定义block
typedef 返回值类型 (^名称)(参数类型...);这是一个block类型,名称就是类型名称
举例: typedef int (MyBlock)(int,int);
MyBlock sumNum1AndNum2 = ^(int num1,int num2){
return num1 + num2;
}
调用:int sum = sumNum1AndNum2(100,200);
MyBlock minusNum1AndNum2 = ^(int num1,num2){
return num1 - num2;
}
调用:int minus = minusNum1AndNum2(200,100);
3,Block如何访问和修改外部变量
首先要明确一点,在block内部引用的变量是局部的还是全局的.
有这样一个例子:
int a = 10;
void (^MyBlock)() = ^{
a = 30;
}
MyBlock();
NSLog(@"a的值是%d",a);
编译一下就会报错,,提示缺少__block,
这是什么原因呢:
如果是局部变量,而且是没有__block修饰的普通局部变量,在block内部访问的话,只是把这个局部变量的值传递进来,所以也不能修改,但是如果是用__block(用static也可以)修饰的局部变量,在block内部访问的话,而是把这个局部变量的地址传递过去了,所以会跟踪这个局部变量的变化,并且可以修改,
如果block内部引用的变量是全局变量的话,那么在block内部访问,他也是把这个变量的地址传递过去了.
4,Block为什么用copy修饰和循环引用问题
因为block创建的时候,它的内存是分配在栈上的(stack),所以如果除了这个作用域他就会被销毁,所以如果在作用域外使用block的话就会崩溃,使用copy修饰block会把block拷贝到堆(heap)上,所以用copy修饰.
如果在block中访问到self的时候一定要格外小心,很有可能造成循环引用的问题,但是也不是绝对,那么什么时候会引起循环引用的问题呢?
比如说有一个类A,在A类中有一个block属性,在控制器中,我们创建A对象,并且把它赋值给一个A类型的属性,再给A的block属性赋值,如果这时候在block代码块中引用了self就会出现一下这种现象,
类A强引用block, 控制器强引用类A, block强引用控制器self, 造成循环引用.
例如:在一个自定义View中有一个文本框(taxtField),在里面输入文字,点击保存按钮就会把输入的文字保存的控制器的label上,
view.h @property (nonatomic,copy)(void (^block)(NSString *)) blockDemo;
- (void)saveText;
view.m中
- (void)saveTexe{
if (self.blockDemo){
self.blockDemo(self.textField.text);
}
}
在viewController中,@property (nonatomic,strong)view viewDemo;
View *viewDemo = [[View alloc] init];
self.viewDemo = viewDemo;
viewDemo.blockDemo = ^(NSString *str){
self.textLabel.text = str;
}
那么如何解决循环引用呢,其实就是使一方变成弱引用就可以了,在这里把block对self的强引用变成弱引用,
__weak typeof(self) weakSelf = self; 使用weakSelf代替self即可.
在ARC中用__weak或者__unsafe_unretain
在非ARC中用__block修饰
5,Block的实际使用
Block 一般是用来表示、简化一小段的程式码,它特别适合用来建立一些同步执行的程式片段、封装一些小型的工作或是用来做为某一个工作完成时的回传呼叫(callback) 。
在新的iOS API中block被大量用来取代传统的delegate和callback,而新的API会大量使用block主要是基于以下两个原因:
可以直接在程式码中撰写等会要接着执行的程式,直接将程式码变成函数的参数传入函数中,这是新API最常使用block的地方。
可以存取区域变数,在传统的callback实作时,若想要存取区域变数得将变数封装成结构才能使用,而block则是可以很方便地直接存取区域变数。
6,Block和代理通知的区别和选择