Block的原理及使用

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和代理通知的区别和选择

你可能感兴趣的:(Block的原理及使用)