Block对变量的引用
如果在 Block 内部使用外部的变量,Block 会持有这个变量。下面来看几中特殊的情况,看 Block 对变量的持有情况如何。
typeof
@interface X : NSObject @property (nonatomic, copy) void(^testBlock1)(void);
@property (nonatomic, copy) void(^testBlock2)(void);
@property (nonatomic, copy) NSString *str; @end @implementation X - (void)test {
self.testBlock1 = ^{
typeof (self) x;
};
} @end
上面代码中, test 方法里面设置了 Block,并且 Block 里面有一句 typeof (self) x,那么在这种情况下,Block 会持有 self 吗?
使用 clang -rewrite-objc 命令,查看 c++ 代码,代码如下:
struct __X__test_block_impl_0 { // Block实现,没有持有 self
struct __block_impl impl;
struct __X__test_block_desc_0* Desc;
__X__test_block_impl_0(void *fp, struct __X__test_block_desc_0 *desc, int flags=0) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __X__test_block_func_0(struct __X__test_block_impl_0 *__cself) { // Block方法实现
X * x; // typeof (self) x;
}
从代码可以看到,typeof (self) x 被替换成了 X *x,因此这种情况下 Block 不会持有 self。
sizeof
将 test 方法修改如下:
- (void)test {
self.testBlock1 = ^{
size_t i = sizeof (self);
};
}
在上面的方法中,Block 内部有使用 sizeof (self),那么 Block 会持有 self 吗?
同样查看 c++ 代码,代码如下:
struct __X__test_block_impl_0 { // Block实现
struct __block_impl impl;
struct __X__test_block_desc_0* Desc;
X *self; // 持有了 self
__X__test_block_impl_0(void *fp, struct __X__test_block_desc_0 *desc, X *_self, int flags=0) : self(_self) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __X__test_block_func_0(struct __X__test_block_impl_0 *__cself) { // Block方法
X *self = __cself->self; // bound by copy
size_t i = sizeof (self);
}
可以看到,在使用 sizeof 时,Block 是持有了 self的。
是否会持有属性
将 test 方法修改如下:
- (void)test {
self.testBlock1 = ^{
self.str = @"KK"; // 设置属性
};
}
@end
上面代码中,Block 除了会持有 self,还会持有属性 str 吗?
查看 c++ 代码,代码如下:
struct __X__test_block_impl_0 { // Block实现
struct __block_impl impl;
struct __X__test_block_desc_0* Desc;
X *self; // 只持有了self,未持有属性str
__X__test_block_impl_0(void *fp, struct __X__test_block_desc_0 *desc, X *_self, int flags=0) : self(_self) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __X__test_block_func_0(struct __X__test_block_impl_0 *__cself) { // Block方法实现
X *self = __cself->self; // bound by copy
((void (*)(id, SEL, NSString *))(void *)objc_msgSend)((id)self, sel_registerName("setStr:"), (NSString *)&__NSConstantStringImpl__var_folders_p6_jy49zvqx2656qb8rbq64hc_w0000gn_T_variable_a3a045_mi_0); // 调用设置属性方法
}
可以看到,Block 没有持有属性 str。
是否会持有实例变量
将 test 方法修改如下:
- (void)test {
self.testBlock1 = ^{
_str = @"KK"; // 调用实例变量
};
}
此时,Block 会持有实例变量吗?
查看 c++ 代码,代码如下:
struct __X__test_block_impl_0 { // Block实现
struct __block_impl impl;
struct __X__test_block_desc_0* Desc;
X *self; // 只持有self,未持有实例变量
__X__test_block_impl_0(void *fp, struct __X__test_block_desc_0 *desc, X *_self, int flags=0) : self(_self) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __X__test_block_func_0(struct __X__test_block_impl_0 *__cself) { // Block方法实现
X *self = __cself->self; // bound by copy
(*(NSString **)((char *)self + OBJC_IVAR_$_X$_str)) = (NSString *)&__NSConstantStringImpl__var_folders_p6_jy49zvqx2656qb8rbq64hc_w0000gn_T_variable_d40340_mi_0; // 使用偏移设置实例变量
}
可以看到,Block 没有持有实例变量。
嵌套Block
将 test 方法修改如下:
- (void)test {
int i = 1;
self.testBlock1 = ^{
self.testBlock2 = ^{ // 嵌套Block
self.str = @"KK";
int j = i;
};
};
}
上面代码中使用了嵌套 Block,内层Block使用了变量 i,而外层 Block 没有使用变量 i,外层 Block 是否会持有变量 i 呢?
查看 c++ 代码,代码如下:
struct __X__test_block_impl_0 { // 内层Block实现
struct __block_impl impl;
struct __X__test_block_desc_0* Desc;
X *self; // 持有了self
int i; // 持有了变量i
__X__test_block_impl_0(void *fp, struct __X__test_block_desc_0 *desc, X *_self, int _i, int flags=0) : self(_self), i(_i) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __X__test_block_func_0(struct __X__test_block_impl_0 *__cself) { // 内层Block 方法实现
X *self = __cself->self; // bound by copy
int i = __cself->i; // bound by copy
((void (*)(id, SEL, NSString *))(void *)objc_msgSend)((id)self, sel_registerName("setStr:"), (NSString *)&__NSConstantStringImpl__var_folders_p6_jy49zvqx2656qb8rbq64hc_w0000gn_T_variable_926b0d_mi_0);
int j = i;
}
struct __X__test_block_impl_1 { // 外层Block实现
struct __block_impl impl;
struct __X__test_block_desc_1* Desc;
X *self; // 持有了self
__X__test_block_impl_1(void *fp, struct __X__test_block_desc_1 *desc, X *_self, int flags=0) : self(_self) {
impl.isa = &_NSConcreteStackBlock;
impl.Flags = flags;
impl.FuncPtr = fp;
Desc = desc;
}
};
static void __X__test_block_func_1(struct __X__test_block_impl_1 *__cself) { // 外层Block方法实现
X *self = __cself->self; // bound by copy
((void (*)(id, SEL, void (*)()))(void *)objc_msgSend)((id)self, sel_registerName("setTestBlock2:"), ((void (*)())&__X__test_block_impl_0((void *)__X__test_block_func_0, &__X__test_block_desc_0_DATA, self, i, 570425344)));
}
从代码上看,内层 Block 持有 self 和变量 i,外层 Block 只是持有了 self,并未持有变量 i。
Block对变量的引用的更多相关文章
- 环境变量、block、修饰符:block对环境变量的引用和修改需要通过修饰符来限定
环境变量.block.修饰符:block对环境变量的引用和修改需要通过修饰符来限定. http://www.cnblogs.com/fengmin/p/5816580.html - (NSUInteg ...
- Block内的强引用
众所周知,当某个对象持有着一个Block的时候,如果在Block内部使用强引用反过来持有这个对象,就会导致引用循环.为了避免引用循环,可以使用__weak修饰符,苹果的官方文档在用代码演示__weak ...
- ARC下的block导致的循环引用问题解析
ARC下的block导致的循环引用问题解析 更详细细节请参考 http://blog.sina.com.cn/s/blog_8c87ba3b0101m599.html ARC下,copy到堆上的blo ...
- 避免Block中的强引用环
[避免Block中的强引用环] In manual reference counting mode, __block id x; has the effect of not retaining x. ...
- Makefile笔记之一 ------ 变量的引用及赋值
1.变量的引用方式: "$(变量名)"或者"¥{变量名}" 例如: ${Objs}就是取变量Objs的值 注意: 当变量名为单字符是可以采用:"$a& ...
- GNU make使用变量⑤变量的引用、定义等
在 Makefile 中,变量是一个名字(像是 C 语言中的宏),代表一个文本字符串(变量的值).在 Makefile 的目标.依赖.命令中引用变量的地方,变量会被它的值所取代(与 C 语言中宏引用的 ...
- Python中的变量、引用、拷贝和作用域
在Python中,变量是没有类型的,这和以往看到的大部分编辑语言都不一样.在使用变量的时候,不需要提前声明,只需要给这个变量赋值即可.但是,当用变量的时候,必须要给这个变量赋值:如果只写一个变量,而没 ...
- 深入理解PHP原理之变量分离/引用
19 Sep 08 深入理解PHP原理之变量分离/引用(Variables Separation) 作者: Laruence( ) 本文地址: http://www.laruence.com/20 ...
- c++变量的引用---5
原创博客:转载请标明出处:http://www.cnblogs.com/zxouxuewei/ 1.引用变量的主要用途: 用作函数的形参,通过将引用变量用作参数,函数将使用原始数据而不是其拷贝. 2. ...
随机推荐
- 什么是Autolayout
Autolayout是一种"自动布局"技术,专门用来布局UI界面的 Autolayout自iOS 6开始引入,由于Xcode 4的不给力,当时并没有得到很大推广 自iOS 7(Xc ...
- Redis sentinel.conf配置文件详解
redis-sentinel.conf配置项说明如下: 1.port 26379 sentinel监听端口,默认是26379,可以修改. 2.sentinel monitor <master-n ...
- Java基础复习(四)
1.Integer与int的区别 int是java提供的8种原始数据类型之一.Java为每个原始类型提供了封装类,Integer是java为int提供的封装类.int的默认值为0,而Integer的默 ...
- Java中Arrays数组工具类的使用全解
本文几乎涵盖了所有的Arrays工具类(基于Java 11)的方法以及使用用例,一站式带你了解Arrays类的用法,希望对大家有帮助. 码字不易,三连支持一下吧 Arrays数组工具类 方法一览表 快 ...
- HTTP状态码100、200、300、400、500、600的含义
1xx (临时响应)表示临时响应并需要请求者继续执行操作的状态代码. 100 (继续) 请求者应当继续提出请求. 服务器返回此代码表示已收到请求的第一部分,正在等待其余部分. 101 (切换协议) 请 ...
- HMS Core机器学习服务助力Zaful用户便捷购物
Zaful是近年来发展良好的出海电商平台,主打快时尚.在洞察其用户有购买街头海报.生活中同款衣物的需求后,Zaful联合HMS Core机器学习服务打造拍照购物服务.用户拍照后可在预设的商品图片库中搜 ...
- [TJOI2013] 奖学金
代码: #include<bits/stdc++.h> using namespace std; long long n,c,ff,ans; long long suma[200010], ...
- 记录一次elasticsearch-5.6.4宕机排查经历
犯罪现场~~ es: 三节点,配置相同 内存: 248G CPU: 没注意看 磁盘: 2T data: 380G左右 indices: 近9800条 在下才疏学浅,目前跟着大佬学习,这个问题还没解决, ...
- Windows查看本机SSH公钥,生成公钥
#Windows查看本机**SSH**公钥,生成公钥<br>--- ### 1.查看 ssh 公钥方法: 1. 打开你的 git bash 窗口 2. 进入 .ssh 目录:cd ~/.s ...
- python3发邮件脚本
官方文档中建议保存token,且token是每2小时更新一次. 所以token先保存在本地token.txt文件夹中,设定计划任务每1小时删除一下token.txt.虽然造成了浪费,对于发消息不多的人 ...