不使用__block修饰:
int num = ;
self.myBlock = ^(){
NSLog(@"block num == %d",num);
}; num = ;
self.myBlock();
NSLog(@"last num == %d",num); // 输入结果:
// 2016-09-13 13:13:15.362 block2[7543:562614] block num == 1
// 2016-09-13 13:13:15.363 block2[7543:562614] last num == 2
num 使用__block修饰后:
__block int num = ;
self.myBlock = ^(){
NSLog(@"block num == %d",num);
}; num = ;
self.myBlock();
NSLog(@"last num == %d",num); // 输出结果:
// 2016-09-13 13:14:20.964 block2[7574:570584] block num == 2
// 2016-09-13 13:14:20.964 block2[7574:570584] last num == 2

如果不用__block修饰变量,在block内强行修改变量会报错:

增加__block后,可以在block内部修改num:

__block int num = ;
self.myBlock = ^(){
num = ;
NSLog(@"block num == %d",num);
}; num = ;
self.myBlock();
NSLog(@"last num == %d",num); // 输出结果:
// 2016-09-13 13:16:44.961 block2[7632:585368] block num == 3
// 2016-09-13 13:16:44.962 block2[7632:585368] last num == 3

block调用完后再修改num:

    __block int num = ;
self.myBlock = ^(){
num = ;
NSLog(@"block num == %d",num);
}; self.myBlock();
num = ;
NSLog(@"last num == %d",num);
// 输出结果:
// 2016-09-13 13:18:38.332 block2[7681:599563] block num == 3
// 2016-09-13 13:18:38.333 block2[7681:599563] last num == 2

block对对象类型的使用和对基本数据类型的使用一样:

不用__block修饰:

    NSString *str = @"";
self.myBlock = ^(){
NSLog(@"block str == %@",str);
};
str = @"";
self.myBlock();
NSLog(@"last str == %@",str); // 输出结果:
// 2016-09-13 13:23:03.091 block2[7815:628689] block str == 1
// 2016-09-13 13:23:03.092 block2[7815:628689] last str == 2

使用__block修饰:

    __block NSString *str = @"";
self.myBlock = ^(){
NSLog(@"block str == %@",str);
};
str = @"";
self.myBlock();
NSLog(@"last str == %@",str); // 输出结果:
// 2016-09-13 13:25:17.199 block2[7867:638529] block str == 2
// 2016-09-13 13:25:17.199 block2[7867:638529] last str == 2

block内部修改对象:

    __block NSString *str = @"";
self.myBlock = ^(){
str = @"";
NSLog(@"block str == %@",str);
};
str = @"";
self.myBlock();
NSLog(@"last str == %@",str); // 输出结果:
// 2016-09-13 13:27:49.440 block2[7937:651966] block str == 3
// 2016-09-13 13:27:49.441 block2[7937:651966] last str == 3

block调用完后再修改对象:

    __block NSString *str = @"";
self.myBlock = ^(){
str = @"";
NSLog(@"block str == %@",str);
};
self.myBlock();
str = @"";
NSLog(@"last str == %@",str); // 输出结果:
// 2016-09-13 13:29:03.705 block2[7973:660957] block str == 3
// 2016-09-13 13:29:03.705 block2[7973:660957] last str == 2

总结:

__block的使用:

block对基本数据类型和对象类型的使用注意是一样的。即:

无论是基本数据类型还是对象类型,如果想在block内部改变外面变量的值,必须用__block修饰;如果外面的值改变也要影响block内部值,也必须用__block修饰。(与block外面的变量的值保持同步)
  即如果想内部改变影响外部或者外部改变影响内部,那么变量必须用__block修饰
  如果不加__block修饰,那么block会捕获变量的值,在之后不受外面变量的改变而改变;并且也不能在block内部修改变量的值
  如果想在block内部修改变量的值,必须用__block修饰

block循环引用:

伴随着block能够捕获变量的能力的一个问题就是, 循环引用, 在ARC中, 只要不是用到纯C语言的库,管理内存的工作都不需要我们完成, 但是循环引用却是我们需要解决的, 最常见的就是当block捕获的变量是一个对象的属性(方法)的时候, 也就是会捕获到self, 那么这个时候就可能会造成循环引用(block属性应该被标记为copy), 解决方法也很简单, 使用一个对self弱引用的指针即可, 这个写法就很多了, 笔者习惯的写法是: __weak typeof(self) weakSelf = self;, 那么在block中使用weakSelf替代self调用相关的属性或者方法, 就不会造成循环引用

weakSelf和strongSelf:

使用weakSelf能够解决block捕获self造成的循环引用的内存泄漏问题, 但是带来的另一个问题就是, 特别是在多线程中,可能在block中代码正在执行的时候, self被销毁了, 因为使用weakSelf捕获到的是self的弱引用, 那么后续的代码就不能够继续执行了, 这个时候为了保证在这个block中self即使被销毁block里面的代码也能正常执行, 我们需要的另一个操作就是, 将weakSelf强引用一次, 让他的引用计数加1, 就能处理这个问题, 就是Apple在wwdc中提到的weak-strong-dance, 笔者习惯的书写方式是: __strong typeof(self) strongSelf = weakSelf;,,, 当然这个必须要明白的是, 这个block里面的strongSelf能够保证里面代码执行完毕的前提是程序能够执行到block, 如果在执行block之前self已经被销毁了, 那么这个block肯定是不会被调用的(block的引用计数已经为0).

block的更多相关文章

  1. Objective-C中block的底层原理

    先出2个考题: 1. 上面打印的是几,captureNum2 出去作用域后是否被销毁?为什么? 同样类型的题目: 问:打印的数字为多少? 有人会回答:mutArray是captureObject方法的 ...

  2. iOS 键盘添加完成按钮,delegate和block回调

    这个是一个比较初级一点的文章,新人可以看看.当然实现这个需求的时候自己也有一点收获,记下来吧. 前两天产品要求在工程的所有数字键盘弹出时,上面带一个小帽子,上面安装一个“完成”按钮,这个完成按钮也没有 ...

  3. python中IndentationError: expected an indented block错误的解决方法

    IndentationError: expected an indented block 翻译为IndentationError:预期的缩进块 解决方法:有冒号的下一行要缩进,该缩进就缩进

  4. JDBC Tutorials: Commit or Rollback transaction in finally block

    http://skeletoncoder.blogspot.com/2006/10/jdbc-tutorials-commit-or-rollback.html JDBC Tutorials: Com ...

  5. 嵌入式&iOS:回调函数(C)与block(OC)传 参/函数 对比

    C的回调函数: callBack.h 1).声明一个doSomeThingCount函数,参数为一个(无返回值,1个int参数的)函数. void DSTCount(void(*CallBack)(i ...

  6. 嵌入式&iOS:回调函数(C)与block(OC)回调对比

    学了OC的block,再写C的回调函数有点别扭,对比下区别,回忆记录下. C的回调函数: callBack.h 1).定义一个回调函数的参数数量.类型. typedef void (*CallBack ...

  7. Block解析(iOS)

    1. 操作系统中的栈和堆 我们先来看看一个由C/C++/OBJC编译的程序占用内存分布的结构: 栈区(stack):由系统自动分配,一般存放函数参数值.局部变量的值等.由编译器自动创建与释放.其操作方 ...

  8. CSS学习笔记——包含块 containing block

    以下内容翻译自CSS 2.1官方文档.网址:https://www.w3.org/TR/CSS2/visudet.html#strut 有时,一个元素的盒子的位置和尺寸根据一个确定的矩形计算,这个确定 ...

  9. 用block做事件回调来简化代码,提高开发效率

       我们在自定义view的时候,通常要考虑view的封装复用,所以如何把view的事件回调给Controller就是个需要好好考虑的问题, 一般来说,可选的方式主要有target-action和de ...

  10. 关于多个block问题

    在某个添加文本的页面中,leftbarbutton是删除(直接将数组中的这个string删除),rightbarbutton是完成,分别对应两个block,完成的block是一开始写的,写到了view ...

随机推荐

  1. 来玩Play框架05 数据库

    作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! 数据库是整个站点的数据储藏室.用户提交的数据可以存储在数据库中,以便未来使用.Pl ...

  2. XTemplate语法基础

    XTemplate 是富逻辑的 KISSY 模板引擎,面向复杂的业务逻辑场景,同时保持高性能和丰富的配置方法,是易学易懂的模板语言. 一个典型的XTemplate模板实例: Hello {{name} ...

  3. android基于口令加密快速搞懂(一)

    import java.util.Random; import javax.crypto.Cipher;import javax.crypto.SecretKey;import javax.crypt ...

  4. pdf文件的导入导出

    下面的代码自己本人没有尝试过,只是用来做记录,用到时候再说! 最近碰见个需求需要实现导出pdf文件,上网查了下代码资料总结了以下代码.可以成功的实现导出pdf文件. 在编码前需要在网上下载个itext ...

  5. YII2 载入默认值 loadDefaultValues

    本人很懒,所以喜欢找现成的东西来用,所以在载入默认值的时候我直接就选择了Yii2 自带的loadDefaultValues 问题来了,我提交的时候发现我在rules里面设置的default没有工作 [ ...

  6. iOS 浅复制和深复制的深层理解,含示例

    转载:https://www.zybuluo.com/MicroCai/note/50592 版权归 @MicroCai 所有 以下是正文: 浅复制就是指针拷贝:深复制就是内容拷贝. 集合的浅复制 ( ...

  7. [笔记]过期的UBUNTU怎么更新软件包

    使用old-releases仓库替换main/security仓库,就像下面这样. sudo sed -i -r 's/([a-z]{2}\.)?archive.ubuntu.com/old-rele ...

  8. SPI基础知识

    Serial Peripheral Interface 是摩托罗拉公司提出的一种总线协议,主要应用在EEPROM,FLASH,实时时钟,A/D转换,以及数字信号处理和数字信号解码器中 是一种高速,全双 ...

  9. linux top命令结果参数详解

    非常详细的top结果说明文档. http://www.cnblogs.com/sbaicl/articles/2752068.html http://bbs.linuxtone.org/forum.p ...

  10. 【Pyrosim案例】01:空气流动

    1 案例说明 Pyrosim是一款FDS前后处理软件.以一个非常简单的案例来描述Pyrosim的工作流程.本案例计算域中包含一个空气入口,一个环境出口. 案例主要演示以下内容: 创建Vents 增加切 ...