BLOCK专题
>>定义并使用一个block 返回值(^名字)(参数列表) =^返回值类型(参数列表){表达式}; 其中返回值和参数列表可以神略 ,最简单的block是 ^{xxxx};
void(^add)(int a , int b ) = ^(int a , int b ){
NSLog(@"a+b=%d",a+b);
};
add(1,3);
>>假设有如下代码
- (void)viewDidLoad {
int f = 0;
void(^add)(int a , int b ) = ^(int a , int b ){
NSLog(@"a+b=%d",a+b);
f = 1;
};
add(1,3);
}
这个时候,会提示f=1;这里出现错误,如果需要修改block以外的局部变量(在这个viewDidLoad函数内的)就需要使用 __block int f = 0;
不需要加入__block就可以直接修改的变量(从下面看出,应该就是变量的生命周期问题)(注释1)
全局变量
类的成员变量
static变量(函数体里面也可以的)
>>有如下代码
int a = 1;
void(^OutPut)() = ^(){
NSLog(@"a=%d",a);
};
a = 2;
OutPut();
上面的代码输出 a=1;
如果修改为 __block int a = 1; 这样会输出 a=2;(所有的可改变的变量都会输出2,在上面注释1处说明的变量)
>>再看一个例子
- (void)viewDidLoad {
NSMutableArray * arr = [NSMutableArray arrayWithObject:@"ddd"];
void(^OutPut)() = ^(){
[arr removeLastObject];
arr = [NSMutableArray new];
};
OutPut();
}
上面提到比如不加入__block的变量是不能在block里面修改的,这里的arr因为没有加__block 所以在执行 arr = [NSMutableArray new];的时候就出错了,但是[arr removeLastObject];并没有任何问题。
>>定义一个 Block Pointer
上面的 void(^add)(inta,int b) 我们其实可以使用一个typedef来实现
typedef void(^AddMethod)(int , int );
AddMethod add = ^(int a , int b ){ NSLog(@“a+b=%d”,a+b};
下面涉及到的内容可以看看如下网址
http://www.cocoachina.com/bbs/read.php?tid=152222
>>NSStackBlock NSGlobalBlock NSMallocBlock 如何区分(可以通过NSLog来打印出一个block属于哪个类型)
NSGlobalBlock:
block内不访问任何变量的
block除了方位全局变量跟static变量的(其中类的成员变量等不在这2种类型里面)
剩下的基本上都是StackBlock
比如:
void(^outPut)() = ^{
NSLog(@"outPut");
};
是GlobalBlock,因为block内部没有访问任何变量
比如:
int a = 0;
void(^outPut)() = ^{
NSLog(@"outPut:%d",a);
};
是StackBlock,因为方位了非(全局,static)变量
比如:
static int a = 0;
void(^outPut)() = ^{
NSLog(@"outPut:%d",a);
};
是GlobalBlock,因为只访问了static变量
>> 有了上面3个Block存储区域的概念,我们就比较容易理解了下面的内容了。
例子:
- void exampleB_addBlockToArray(NSMutableArray *array) {
- char b = 'B';
- [array addObject:^{
- printf("%c\n", b);
- }];
- }
- void exampleB() {
- NSMutableArray *array = [NSMutableArray array];
- exampleB_addBlockToArray(array);
- void (^block)() = [array objectAtIndex:0];
- block();
- }
首先看addObject这个block是一个StackBlock
在ARC情况下,执行正确 :因为ARC下会自动将StackBlock拷贝到MallocStack中
在非ARC情况下,执行错误:因为在exampleB_addBlockToArray函数返回之后,这个block失效了
例子
- void exampleC_addBlockToArray(NSMutableArray *array) {
- [array addObject:^{
- printf("C\n");
- }];
- }
- void exampleC() {
- NSMutableArray *array = [NSMutableArray array];
- exampleC_addBlockToArray(array);
- void (^block)() = [array objectAtIndex:0];
- block();
- }
首先看这里的addObject这个block没有访问任何的外部变量,所以这是一个GlobalBlock,不管在ARC还是非ARC下,都是正确的。
>> 为什么在使用property属性修饰block的时候,需要用到copy操作?
在非ARC下,需要将block从栈中copy到堆中,避免被自动释放。对block 进行retain操作是无效的
在ARC下,block会自动copy到堆中。
>> Block循环引用问题(一定要是双方的)
使用block的时候,经常会由于循环引用的问题导致内存无法释放,(对象拥有block,block又用到对象)
比如
self.myBlock = ^{ [self xxxxx];};
这里myBlock是self的属性,而在myBlock里面又使用到了self;self在释放的时候必定要先释放myBlock,但是myBlock需要先释放里面的self,这样就导致死循环了。
要打破这个格局,可以使用如下
在ARC下:
__weak typeof(self) weakSelf = self;
self.myBlock = ^{ [weakSelf xxxxx];};
如果在block里面会多次用到weakSelf 那应该使用如下的strongSelf
__weak typeof(self) weakSelf = self;
self.myBlock = ^{
__strong typeof(self)strongSelf = weakSelf;
[strongSelf xxxxx];
[strongSelf bbbbb];
};
在非ARC下,直接将__weak改为__block就可以(__block typeof(self)weakSelf=self;)。
所以__block的意思在ARC下只表示可修改;在非ARC下,表示可修改、不要retain这个变量.
>> 在一个类中,并不是直接使用了self才会导致强引用self的。比如类中有一个实例变量 int value;
那么在 dispatch_async(queue,^{ doSomthingWith(value); } ); 这个时候用到了类的实例变量value,实际上也是强应用了 self。如果queue是类的成员变量,那就会出现死循环的问题。如果是这样 dispatch_async(dispatch_get_global_queue(0,0),^{ doSomthingWith(value); } ); 这个就是单向引用,不会造成死循环。
>> block与函数的区别
block是对象,它封装了一段代码,这段代码可以在任何时候执行。block可以作为函数参数或者函数的返回值,而其本身又可以带输入参数或者返回值。
block跟传统的函数指针很类似,但是block是内联的(inline),并且它对局部变量只是只读的。
block可以访问局部变量,但是不能修改
在需要修改的局部变量前加入__block
定义一个block看上去像定义一个函数差不多,void(^bbb)(void)=^(int a){…..};
但是还是有很大的区别的,block的定义可以在一个函数的内部实现。
>> __block 与 __weak的区别
__block 在ARC以及MRC模式下都可以使用。
__weak是ARC模式下使用的,代表的是弱引用,防止循环引用。
__block在非ARC下表示可以修改、如果修饰的是对象,表示不要retain(防止循环引用)
__block对象在block中可以被重新赋值,__weak不可以。
BLOCK专题的更多相关文章
- ios专题 -block用法
what is block Blocks are a language-level feature added to C, Objective-C and C++, which allow you t ...
- [.NET领域驱动设计实战系列]专题八:DDD案例:网上书店分布式消息队列和分布式缓存的实现
一.引言 在上一专题中,商家发货和用户确认收货功能引入了消息队列来实现的,引入消息队列的好处可以保证消息的顺序处理,并且具有良好的可扩展性.但是上一专题消息队列是基于内存中队列对象来实现,这样实现有一 ...
- Block Chain, a protocol view
我做了个区块链的文档,给自己扫盲用的,有兴趣的可以看下,主要是自己画示意图比较好理解,示意图之后的专题部分,内容直接取自参考链接.网上的资料都是谈区块链有什么性质.有什么能力.有什么应用之类的,我主要 ...
- block的内部实现
主要内容: 一.block相关的题目 二.block的定义 三.block的实现 四.捕获自动变量值 五.block存储区域 六.截获对象 一.block相关的题目 这是一篇比较长的博文,前部分是bl ...
- 【转】css学习专题-BFC
css学习专题-BFC 转自:原文链接:css学习专题-BFC文章目录 BFC:Block Formatting Context. BFC就是一种布局方式,在这种布局方式下,盒子们自所在的 conta ...
- HBase 专题技术收录
HBase系列: 博客地址:http://www.cnblogs.com/panfeng412/tag/HBase/ 技术专题文章: HBase中MVCC的实现机制及应用情况 HBase在单Colum ...
- CSS浮动专题!
在css中,浮动问题可能是很多刚入门的小白比较头疼的问题. 1,首先先来介绍一下两种浮动类型:左浮动和右浮动 1) float:left;左浮动,后面的内容会流向对象的右侧 2) float:righ ...
- mysql优化专题」90%程序员都会忽略的增删改优化(2)
补充知识点:操作数据语句优化的认识 通常情况下,当访问某张表的时候,读取者首先必须获取该表的锁,如果有写入操作到达,那么写入者一直等待读取者完成操作(查询开始之后就不能中断,因此允许读取者完成操作). ...
- Oracle之SQL优化专题01-查看SQL执行计划的方法
在我2014年总结的"SQL Tuning 基础概述"中,其实已经介绍了一些查看SQL执行计划的方法,但是不够系统和全面,所以本次SQL优化专题,就首先要系统的介绍一下查看SQL执 ...
随机推荐
- 重写equals()方法时,需要同时重写hashCode()方法
package com.wangzhu.map; import java.util.HashMap; /** * hashCode方法的主要作用是为了配合基于散列的集合一起正常运行,<br/&g ...
- C语言中volatile关键字的作用
http://blog.csdn.net/tigerjibo/article/details/7427366#comments 一.前言 1.编译器优化介绍: 由 于内存访问速度远不及CPU处理速度, ...
- Win32 SDK程序创建一些控件(简单调用InitCommonControlsEx,并指定ICC_LISTVIEW_CLASSES控件就可以了)
在Win32 SDK中创建一些控件的时候需要注意一下(具体是哪些控件请参看MSDN文档中列出来的) /* MSDN:Carries information used to load common co ...
- Spring的父子容器问题
在ssm框架搭建的时候 配置了一个Spring容器,又配置了一个前端控制器 <!-- 初始化spring容器 --> <context-param> <param-nam ...
- NSDictionary 键值对查找
NSDictionary *dic1=[NSDictionary dictionaryWithObjectsAndKeys: @"1",@"a", ...
- 4、JPA table主键生成策略(在JPA中table策略是首推!!!)
用 table 来生成主键详解 它是在不影响性能情况下,通用性最强的 JPA 主键生成器.这种方法生成主键的策略可以适用于任何数据库,不必担心不同数据库不兼容造成的问题. initialValue不起 ...
- IOS判断手机型号
#define iPhone5 ([UIScreen instancesRespondToSelector:@selector(currentMode)] ? CGSizeEqualToSize(CG ...
- gulp.watch监听文件
Gulp.watch()会返回我们熟知的watcher.我们可以利用watcher来监听额外的事件或者向watch中添加文件. 例如,在执行一系列任务和调用一个函数时,你就可以在返回的watcher中 ...
- Hadoop集群(第7期)_Eclipse开发环境设置
1.Hadoop开发环境简介 1.1 Hadoop集群简介 Java版本:jdk-6u31-linux-i586.bin Linux系统:CentOS6.0 Hadoop版本:hadoop-1.0.0 ...
- hdu 3501 Calculation 2 (欧拉函数)
题目 题意:求小于n并且 和n不互质的数的总和. 思路:求小于n并且与n互质的数的和为:n*phi[n]/2 . 若a和n互质,n-a必定也和n互质(a<n).也就是说num必定为偶数.其中互质 ...