dispatch_async 的 block 中是否该使用_weak self
问题分析
我看过很多文章关于在dispatch_async的block里面使用_weak self, 但是让我疑惑的是,以下代码是否需要必须使用_weak self, 因为我也看到了很多观点说,在有些情况下不需要使用__weak self.
self.myQueue = dispatch_queue_create("com.biview.core_data", NULL);
dispatch_async(self.myQueue, ^(void){
if(!self.var1){
self.var1 =
}
dispatch_async(dispatch_get_main_queue(), ^(void){
if([self.var2 superview]) {
[self.var2 removeFromSuperview];
}
[self.Label setText:text];
});
});复制代码
解析
针对上面的问题,我们假设:self是指向UIViewController的对象指针。
考虑以下几点:
- UIViewController是"UIkit"对象,UIKit对象不应该在非主线程发送消息,也就是说,这些方法只能在主线程中执行。
- 当一个block被添加进一个同步或者异步队列,这个block最终都会被执行,除非在执行到它之前应用程序被杀死。
- 当block被拷贝的时候,strong类型的指针会被retained, 当block执行完毕之后被销毁的时候才会执行released操作。
- weak类型的指针不会被retained和released。
在上面的例子中,self是在主线程的队列中,不必担心有任何bug产生。
究竟发生了什么?
当在dispatch的异步队列的block中捕获到self时,self会被执行retained操作,当block执行完毕后self执行released操作。
这意味着:当block执行完毕后,self的生命周期才会结束。上例中的第二个block是在主线程的队列中,它保证了self一直存活着当这个block被执行的时候。
在程序中存在潜在危险的操作是:延长 self 的生命周期。
如果你明确的不希望延长UIViewController对象的生命周期,而是当block被执行的时候去检查UIViewController对象到底是否存在,你可以使用 _weak self. 需要注意的是block最后都会被执行,不管UIViewController是否存活还是已经被释放了。
如果你希望如果UIViewController已经被释放了,那么block不做任何事情,可以写成 _weak self.
MyController * _weak weakSelf = self;
dispatch_async(queue, ^{
MyController *strongSelf = weakSelf;
if(strongSelf){
...
}else {
// self has been deallocted in the meantime.
}
});复制代码
不能在非主线程中向UIKit对象发送消息。
另一个细微的错误可能发生在UIKit对象执行方法在非主线程。
如果block在异步线程中捕获了一个UIKit对象,可能发生的是:block 是最后一个持有改UIKit的强引用。当block执行完的时候,UIKit对象将被release,因为是UIKit对象的最后一个强引用,所有该UIKit对象将被释放,但是,释放操作发生在block所执行的线程-它不是主线程,所有,风险即将发生,UIKit对象的dealloc方法将被调用(UI 对象应该在主线程中被回收,因为在它们的 dealloc 方法被调用回收的时候,可能会去改变 view 的结构关系,而如我们所知,这种操作应该放在主线程来进行,见参考二)。
避免这个错误:
UIViewController *strongUIKitPointer = ...
dispatch_async(non_main_queue), ^{
...//do someting
dispatch(dispatch_get_main_queue(),^{
[strongUIkitPointer self]; //self is a method, too -doing nothing.
});
});复制代码
举例
双向强引用发生在:一个强类型对象A持有一个强类型对象B,并且对象B强引用对象A。“Block”是一个强引用对象。
人为的双向强引用举例:
typedef void(^my_completion_block_t)(NSArray* result);
@interface UserViewController : UIViewController
@property (nonatomic, copy) my_completion_block_t completion;
@property (nonatomic) NSArray *users;复制代码
on "UserViewController.m"
self.completion = ^(NSArray *users){
self.users = users;
}
[self fetchUsers];复制代码
这是一个典型了强引用循环。UserViewController 有一个Block类型的属性,所有UserViewController对象强引用着block。而block捕获到self的时候执行强引用操作,所有形成了强引用循环。
解决方式:
使用_weak 指针指向self.
UserViewController * _weak weakSelf = self;
self.completion = ^(NSArray *user){
UsersViewController *strongSelf = weakSelf;
if(strongSelf){
strongSelf.users = users;
}else{
// the view controller does not exist anymore.
}
};复制代码使用block 指针执行self, 执行完毕后将block 指针指向nil.
UsersViewController *__block blockSelf = self;
self.completion=^(NSArayy *users){
self.completion = ^(NSArray *users){
blockSelf.users = users;
blockSelf = nil;
}
}复制代码
--完--
参考
dispatch_async 的 block 中是否该使用_weak self的更多相关文章
- @weakify, @strongify ObjC的Block中使用weakSelf/strongSelf @weakify/@strongify
首先要说说什么时候使用weakSelf和strongSelf. 下面引用一篇博客<到底什么时候才需要在ObjC的Block中使用weakSelf/strongSelf>的内容: Objec ...
- ObjC的Block中使用weakSelf/strongSelf @weakify/@strongify
首先要说说什么时候使用weakSelf和strongSelf. 下面引用一篇博客<到底什么时候才需要在ObjC的Block中使用weakSelf/strongSelf>的内容: Objec ...
- iOS Block中的weakSelf/strongSelf
Objective C 的 Block 是一个很实用的语法,特别是与GCD结合使用,可以很方便地实现并发.异步任务.但是,如果使用不当,Block 也会引起一些循环引用问题(retain cycle) ...
- 到底什么时候才需要在ObjC的Block中使用weakSelf/strongSelf
转载,原文: http://blog.lessfun.com/blog/2014/11/22/when-should-use-weakself-and-strongself-in-objc-block ...
- iOS 中block中使用了外部变量的分析
例子1: ; void (^blk)(void) = ^(){ printf("in block %d[%p]\n", val, &val); //in block 10[ ...
- block中如何避免循环引用
使用 weak–strong dance 技术 block 可以直接引用 self,但是要非常小心地在 block 中引用 self.因为在 block 引用 self,可能会导致循环引用.如下例所示 ...
- ios 使用block中使用self可能产生的循环引用
在block中调用 self,那么就会引起循环引用问题,那么这是为什么呢? 为什么self会对block进行强引用呢???? 这里推荐一篇关于block的专业文章,http://blog.csdn.n ...
- block中出现此种报错: Incompatible block pointer types initializing 'float (^__strong)(float, float)' with an expression of type 'int (^)(float, float)'
当block(代码块)的返回值是float时,应注意的地方:定义的返回值类型一定要与return的返回值类型一样 我们以两个数的四则运算来举例 在main.m文件中的四则运算中,我采用两种返回值类型( ...
- Block中的引用循环
原文地址:http://www.cnblogs.com/lujianwenance/p/5910490.html Block在实际的开发中非常的常用,事件回调.传值.封装成代码块调用等等.很多人都对b ...
随机推荐
- PTA数据结构与算法题目集(中文) 7-11
PTA数据结构与算法题目集(中文) 7-11 7-11 关键活动 (30 分) 假定一个工程项目由一组子任务构成,子任务之间有的可以并行执行,有的必须在完成了其它一些子任务后才能执行.“任务调度 ...
- Oracle 数据库创建导入
Oracle 数据库创建导入 由 Alma 创建, 最后一次修改 2018-06-04 14:37:50 在本章教程中,将教大家如何在Oracle 中创建导入数据库. 注意:本教程中的有些命令您可能并 ...
- Centos网络的配置
...
- android之间的各项信息传输类型
首先是activity想fragment怎样动态的传输数据: 一:activity与fragment之间进行数据传递是,在Activity中将要传递的数据封装在一Bundle中,使用setArgume ...
- 在tap的碎片上与活动进行绑定实现点击事件(日期时间选择以及按钮跳转时间)
主要是掌握怎样在Fragment类型的.java文件中实现对于文本框或者按钮点击事件的触发操作. 相应的出发时间都是之前的代码.主要是怎样在Fragment怎样实现相应的操作主要是对于getActiv ...
- Linux网络安全篇,认识防火墙(一)
一.概念 防火墙分为软件防火墙和硬件防火墙.我们的主要讨论范围为软件防火墙. 软件防火墙又分为网络型和单一型的管理. 1.单一主机型防火墙 (1)数据包过滤型的Netfilter (2)依据服务软件程 ...
- "无用的文本"组件:<s> —— 快应用组件库H-UI
 <import name="s" src="../Common/ui/h-ui/text/c_tag_del"></import> ...
- alg-最长公共子串
class Solution { public: const std::vector<std::string> LongestCommonSubstring(const std::stri ...
- docker-compose 基于Dockerfile 安装并启动redis容器的血案
前言 为了实现"一键部署"的目的,我采用Dockerfile 和 docker-compose来实现自己的目的.这个过程中,我怎么也无法启动自己的redis-server服务. 目 ...
- AJ学IOS 之微博项目实战(7)程序启动新特性用UICollectionViewController实现
AJ分享,必须精品 一:效果 这里实现了大多数app都会有的软件新特性的功能,用的是UICollectionViewController实现的 二:思路 这里用了UICollectionViewCon ...