capturing self strongly in this block is likely to lead to a retain cycle 警告解决
警告:
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 警告解决相关推荐
- Capturing 'self' strongly in this block is likely to lead to a retain cycle
技术分享 » Phone | 阅读(1629) | 评论(0) May 10 2014 _player.completionBlock = ^{ [self stopPlay] ...
- iOS-bug Capturing ‘self‘ strongly in this block is likely to lead to a retain cycle
__weak typeof(self) weakSelf = self; 再把警告部分的self改成weakself即可
- 喜欢用Block的值得注意-Block的Retain Cycle的解决方法
本文不讲block如何声明及使用,只讲block在使用过程中暂时遇到及带来的隐性危险. 主要基于两点进行演示: 1.block 的循环引用(retain cycle) 2.去除block产生的告警时, ...
- HDFS block丢失过多进入安全模式(safe mode)的解决方法
1.背景 hdfs 重启hdfs,发现HA集群总是nameNode 主备挂掉,然后报错 Safe mode is ON. The number of live datanodes 5 has reac ...
- Block相关内容梳理
什么是block Block是将函数及其上下文封装起来的对象. 源码分析 编译器是如何实现block的? 新建Objective-C文件命名为MyClass,在.m文件中实现如下代码: #import ...
- 和block循环引用说再见
to be block? or to be delegate? 这是一个钻石恒久远的问题.个人在编码中暂时没有发现两者不能通用的地方,习惯上更偏向于block,没有什么很深刻的原因,只是认为block ...
- 读书笔记(UIKit性能优化解析和Block经典实用)(一)
读书笔记(一) 一.instancetype 和 id 作为初始化实例 返回值的不同 Objective-C的一些使用惯例不仅仅是好的编程习惯,更是给编译器的隐藏指令. 例如, alloc 和 ini ...
- iOS:消除项目中警告
引言: 在iOS开发过程中, 我们可能会碰到一些系统方法弃用, weak.循环引用.不能执行之类的警告. 有代码洁癖的孩子们很想消除他们, 今天就让我们来一次Fuck 警告!! 首先学会基本的语句: ...
- iOS OC消除黄色警告⚠️ (不断的更新中...)
开发一个项目时,难免会产生很多警告,一些是第三方或是老代码不再被支持造成的,但并不影响使用,这些警告其实可以直接隐藏掉!还有一些警告可能是系统方法弃用.不兼容指针类型.未使用变量.未使用default ...
- Objective-C Blocks Caveat
http://albertodebortoli.github.io/blog/2013/08/03/objective-c-blocks-caveat/ I like blocks. I really ...
最新文章
- python3.7和3.5_Ubuntu更新python3.5到python3.7
- Vista初级使用技巧及故障总结
- python数据结构-图
- C语言实现井子格游戏
- python学习一:基本数据类型
- 知识点:计算机网络的 89 个核心概念
- 【虹科免费直播预告】光电技术直播月重磅来袭!
- 高通工具过滤_高通QXDM|高通诊断监视工具(Qualcomm QXDM)下载v3.14 官方版 - 欧普软件下载...
- sql和mysql 语法区别吗_sql和mysql语法有什么不同
- 泰坦尼克 数据集_Kaggle-泰坦尼克-学习心得(高分容易,理解很难)——第1篇...
- 360浏览器保存的html没有图标,电脑360浏览器图标不见了怎么办
- SOFTICE 初使用
- 理财通app的设计与实现(六)
- 互联网金融项目数据分析
- 【Ruby on Rails全栈课程】4.1 点赞功能
- HP 孙振耀 九大感言(转)
- 中国专网通信行业投资布局与运营前景能力分析报告2022年
- linux下远程桌面客户端Remmina
- SQL 语句中单引号、双引号的用法
- [计划安排] 408经验帖(转自某大神)