警告:

capturing self strongly in this block is likely to lead to a retain cycle

意思是block会retain一次,所以使用前最好

__block ViewController *strongBlock = self;

    __block ViewController *strongBlock = self;

简单代码如下:

    self.tableViewDelegate.didSelectedBlock = ^(UITableView *tableview, NSIndexPath *indexPath) {MessageViewController *messageVC = [[MessageViewController alloc]init];[self presentViewController:messageVC animated:YES completion:nil];};

修改后的代码:

    __block ViewController *strongBlock = self;self.tableViewDelegate.didSelectedBlock = ^(UITableView *tableview, NSIndexPath *indexPath) {MessageViewController *messageVC = [[MessageViewController alloc]init];[self presentViewController:messageVC animated:YES completion:nil];};

这样就没有警告了!!!

__block关键字:

根据内存地址变化可见,__block所起到的作用就是只要观察到该变量被 block 所持有,就将“外部变量”在栈中的内存地址放到了堆中。进而在block内部也可以修改外部变量的值。原先地址是否直接抛弃不用再继续研究.

Block不允许修改外部变量的值,Apple这样设计,应该是考虑到了block的特殊性,block也属于“函数”的范畴,变量进入block,实际就是已经改变了作用域。在几个作用域之间进行切换时,如果不加上这样的限制,变量的可维护性将大大降低。又比如我想在block内声明了一个与外部同名的变量,此时是允许呢还是不允许呢?只有加上了这样的限制,这样的情景才能实现。

编译器做了什么?

一般使用的话,到这个程度已经足够了。我们已经知道了加上__block关键字之后,编译器通过将外部变量同block一起copy到了堆区,并且将“外部变量”在栈中的内存地址改为了堆中的新地址。
如果多问一个为什么?编译器是怎么做到这样的呢?我们通过clang将 OC 代码转换为 C++ 文件:

clang -rewrite-objc 源代码文件名

转译的时候遇到了几个问题:

#import <UIKit/UIKit.h>
**        ^**
1 error generated.

通过Objective-C编译成C++代码报错文中的方式可以转译,但是又出现了新的问题;
2.clang: warning: using sysroot for 'iPhoneSimulator' but targeting 'MacOSX'
这个问题没能解决,然后换了个思路转译,
新代码如下

//坑爹的是NSLog都不能使用,不然会报NSLog错误。说白了还是工具不熟悉,为什么会出现这个情况都不清楚。有机会再看吧
int main() {int  a = 1;void(^testBlock)(void) = ^(void){};testBlock();__block int b = 2;void(^testBlockb)(void) = ^(void){b = 3;};testBlockb();return 0;
}

转译后代码如下

int main() {int a = 1;void(*testBlock)(void) = ((void (*)())&__main_block_impl_0((void *)__main_block_func_0, &__main_block_desc_0_DATA));((void (*)(__block_impl *))((__block_impl *)testBlock)->FuncPtr)((__block_impl *)testBlock);__attribute__((__blocks__(byref))) __Block_byref_b_0 b = {(void*)0,(__Block_byref_b_0 *)&b, 0, sizeof(__Block_byref_b_0), 2};void(*testBlockb)(void) = ((void (*)())&__main_block_impl_1((void *)__main_block_func_1, &__main_block_desc_1_DATA, (__Block_byref_b_0 *)&b, 570425344));((void (*)(__block_impl *))((__block_impl *)testBlockb)->FuncPtr)((__block_impl *)testBlockb);return 0;
}

代码变的很长很长,我们的目的是研究__block加上去之后编译器的操作,精简下就是

//加__block前的声明变量是这样的
int a = 1;
//加__block后的声明变量是这样的
__attribute__((__blocks__(byref))) __Block_byref_b_0 b = {(void*)0,(__Block_byref_b_0 *)&b, 0,sizeof(__Block_byref_b_0), 2};

可以看到增加了__block修饰之后,编译器做了不少工作,修饰词中有__Block_byref_b_0重复出现,这是一个与block一样的结构体类型的自动变量实例!!!!
此时我们在block内部访问val变量则需要通过一个叫__forwarding的成员变量来间接访问val变量。
讲__forwarding之前,需要先讨论一下block的存储域及copy操作。

1.Block的存储域及copy操作

前面提到,block内部的作用域是在堆上的,并且调用变量时会将变量copy到堆上,那么block本身是存储在堆上还是栈上呢?
我们先来看看一个由C/C++/OBJC编译的程序占用内存分布的结构:

实际上,block有三种类型,

  • 全局块(_NSConcreteGlobalBlock)
  • 栈块(_NSConcreteStackBlock)
  • 堆块(_NSConcreteMallocBlock)
    这三种block各自的存储域如下图:

    三种block各自的存储域

    简而言之,存储在栈中的Block就是栈块、存储在堆中的就是堆块、既不在栈中也不在堆中的块就是全局块。(这听起来似乎与文章上半部分的说明有冲突呢?其实并不然)

那么,我们如何判断这个block的存储位置呢?
(1)Block不访问外界变量(包括栈中和堆中的变量)
Block 既不在栈又不在堆中,在代码段中,ARC和MRC下都是如此。此时为全局块。(_NSConcreteGlobalBlock)
(2)Block访问外界变量
MRC 环境下:访问外界变量的 Block 默认存储栈中。
ARC 环境下:访问外界变量的 Block 默认存储在堆中(实际是放在栈区,然后ARC情况下自动又拷贝到堆区),自动释放。

ARC下,访问外界变量的 Block为什么要自动从栈区拷贝到堆区呢?
栈上的Block,如果其所属的变量作用域结束,该Block就被废弃,如同一般的自动变量。当然,Block中的__block变量也同时被废弃。如下图:

栈上的block的生命周期

为了解决栈块在其变量作用域结束之后被废弃(释放)的问题,我们需要把Block复制到堆中,延长其生命周期。开启ARC时,大多数情况下编译器会恰当地进行判断是否有需要将Block从栈复制到堆,如果有,自动生成将Block从栈上复制到堆上的代码。Block的复制操作执行的是copy实例方法。Block只要调用了copy方法,栈块就会变成堆块。
如下图:

block的copy操作原理

在非ARC情况下则需要开发者调用copy方法手动复制,由于开发中几乎都是ARC模式,所以手动复制内容不再过多研究。
将Block从栈上复制到堆上相当消耗CPU,所以当Block设置在栈上也能够使用时,就不要复制了,因为此时的复制只是在浪费CPU资源。
Block的复制操作执行的是copy实例方法。不同类型的Block使用copy方法的效果如下表:

根据表得知,Block在堆中copy会造成引用计数增加,这与其他Objective-C对象是一样的。虽然Block在栈中也是以对象的身份存在,但是栈块没有引用计数,因为不需要,我们都知道栈区的内存由编译器自动分配释放。
不管Block存储域在何处,用copy方法复制都不会引起任何问题。在不确定时调用copy方法即可。

在ARC有效时,多次调用copy方法完全没有问题:

blk = [[[[blk copy] copy] copy] copy];
// 经过多次复制,变量blk仍然持有Block的强引用,该Block不会被废弃。

2.__block变量与__forwarding

在copy操作之后,既然__block变量也被copy到堆上去了, 那么访问该变量是访问栈上的还是堆上的呢?__forwarding 终于要闪亮登场了,如下图:

通过__forwarding, 无论是在block中还是 block外访问__block变量, 也不管该变量在栈上或堆上, 都能顺利地访问同一个__block变量。
值得注意的是,在ARC下,使用 __block 也有可能带来的循环引用,

capturing self strongly in this block is likely to lead to a retain cycle 警告解决相关推荐

  1. Capturing 'self' strongly in this block is likely to lead to a retain cycle

    技术分享 » Phone | 阅读(1629) | 评论(0) May 10 2014 _player.completionBlock = ^{             [self stopPlay] ...

  2. iOS-bug Capturing ‘self‘ strongly in this block is likely to lead to a retain cycle

    __weak typeof(self) weakSelf = self; 再把警告部分的self改成weakself即可

  3. 喜欢用Block的值得注意-Block的Retain Cycle的解决方法

    本文不讲block如何声明及使用,只讲block在使用过程中暂时遇到及带来的隐性危险. 主要基于两点进行演示: 1.block 的循环引用(retain cycle) 2.去除block产生的告警时, ...

  4. HDFS block丢失过多进入安全模式(safe mode)的解决方法

    1.背景 hdfs 重启hdfs,发现HA集群总是nameNode 主备挂掉,然后报错 Safe mode is ON. The number of live datanodes 5 has reac ...

  5. Block相关内容梳理

    什么是block Block是将函数及其上下文封装起来的对象. 源码分析 编译器是如何实现block的? 新建Objective-C文件命名为MyClass,在.m文件中实现如下代码: #import ...

  6. 和block循环引用说再见

    to be block? or to be delegate? 这是一个钻石恒久远的问题.个人在编码中暂时没有发现两者不能通用的地方,习惯上更偏向于block,没有什么很深刻的原因,只是认为block ...

  7. 读书笔记(UIKit性能优化解析和Block经典实用)(一)

    读书笔记(一) 一.instancetype 和 id 作为初始化实例 返回值的不同 Objective-C的一些使用惯例不仅仅是好的编程习惯,更是给编译器的隐藏指令. 例如, alloc 和 ini ...

  8. iOS:消除项目中警告

    引言: 在iOS开发过程中, 我们可能会碰到一些系统方法弃用, weak.循环引用.不能执行之类的警告. 有代码洁癖的孩子们很想消除他们, 今天就让我们来一次Fuck 警告!! 首先学会基本的语句: ...

  9. iOS OC消除黄色警告⚠️ (不断的更新中...)

    开发一个项目时,难免会产生很多警告,一些是第三方或是老代码不再被支持造成的,但并不影响使用,这些警告其实可以直接隐藏掉!还有一些警告可能是系统方法弃用.不兼容指针类型.未使用变量.未使用default ...

  10. Objective-C Blocks Caveat

    http://albertodebortoli.github.io/blog/2013/08/03/objective-c-blocks-caveat/ I like blocks. I really ...

最新文章

  1. python3.7和3.5_Ubuntu更新python3.5到python3.7
  2. Vista初级使用技巧及故障总结
  3. python数据结构-图
  4. C语言实现井子格游戏
  5. python学习一:基本数据类型
  6. 知识点:计算机网络的 89 个核心概念
  7. 【虹科免费直播预告】光电技术直播月重磅来袭!
  8. 高通工具过滤_高通QXDM|高通诊断监视工具(Qualcomm QXDM)下载v3.14 官方版 - 欧普软件下载...
  9. sql和mysql 语法区别吗_sql和mysql语法有什么不同
  10. 泰坦尼克 数据集_Kaggle-泰坦尼克-学习心得(高分容易,理解很难)——第1篇...
  11. 360浏览器保存的html没有图标,电脑360浏览器图标不见了怎么办
  12. SOFTICE 初使用
  13. 理财通app的设计与实现(六)
  14. 互联网金融项目数据分析
  15. 【Ruby on Rails全栈课程】4.1 点赞功能
  16. HP 孙振耀 九大感言(转)
  17. 中国专网通信行业投资布局与运营前景能力分析报告2022年
  18. linux下远程桌面客户端Remmina
  19. SQL 语句中单引号、双引号的用法
  20. [计划安排] 408经验帖(转自某大神)

热门文章

  1. oracle统计个数函数,oracle中字符串统计的函数
  2. 【程序员必读】经验:编程的智慧
  3. Ubuntu下安装anydesk、realVNC实现远程
  4. vue 生命周期图 + activated + deactivated
  5. lamp mysql什么意思_lamp是什么意思
  6. 深度学习-训练集图片输入神经网络前的标准化(附代码)
  7. 启动maven nexus3时遇到的诡异异常
  8. 如何提升Excel水平?5个Excel常用技巧,学会不加班
  9. 用php上传头像的步骤,php怎么上传头像
  10. day05匿名函数,内置函数,二分法,递归,模块