问题分析

我看过很多文章关于在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的时候执行强引用操作,所有形成了强引用循环。

解决方式:

  1. 使用_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.
    }
    };复制代码
  2. 使用block 指针执行self, 执行完毕后将block 指针指向nil.

    UsersViewController *__block blockSelf = self;
    self.completion=^(NSArayy *users){
    self.completion = ^(NSArray *users){
    blockSelf.users = users;
    blockSelf = nil;
    }
    }复制代码

--完--

参考

  1. stack overflow
  2. objc-cn

dispatch_async 的 block 中是否该使用_weak self的更多相关文章

  1. @weakify, @strongify ObjC的Block中使用weakSelf/strongSelf @weakify/@strongify

    首先要说说什么时候使用weakSelf和strongSelf. 下面引用一篇博客<到底什么时候才需要在ObjC的Block中使用weakSelf/strongSelf>的内容: Objec ...

  2. ObjC的Block中使用weakSelf/strongSelf @weakify/@strongify

    首先要说说什么时候使用weakSelf和strongSelf. 下面引用一篇博客<到底什么时候才需要在ObjC的Block中使用weakSelf/strongSelf>的内容: Objec ...

  3. iOS Block中的weakSelf/strongSelf

    Objective C 的 Block 是一个很实用的语法,特别是与GCD结合使用,可以很方便地实现并发.异步任务.但是,如果使用不当,Block 也会引起一些循环引用问题(retain cycle) ...

  4. 到底什么时候才需要在ObjC的Block中使用weakSelf/strongSelf

    转载,原文: http://blog.lessfun.com/blog/2014/11/22/when-should-use-weakself-and-strongself-in-objc-block ...

  5. iOS 中block中使用了外部变量的分析

    例子1: ; void (^blk)(void) = ^(){ printf("in block %d[%p]\n", val, &val); //in block 10[ ...

  6. block中如何避免循环引用

    使用 weak–strong dance 技术 block 可以直接引用 self,但是要非常小心地在 block 中引用 self.因为在 block 引用 self,可能会导致循环引用.如下例所示 ...

  7. ios 使用block中使用self可能产生的循环引用

    在block中调用 self,那么就会引起循环引用问题,那么这是为什么呢? 为什么self会对block进行强引用呢???? 这里推荐一篇关于block的专业文章,http://blog.csdn.n ...

  8. block中出现此种报错: Incompatible block pointer types initializing 'float (^__strong)(float, float)' with an expression of type 'int (^)(float, float)'

    当block(代码块)的返回值是float时,应注意的地方:定义的返回值类型一定要与return的返回值类型一样 我们以两个数的四则运算来举例 在main.m文件中的四则运算中,我采用两种返回值类型( ...

  9. Block中的引用循环

    原文地址:http://www.cnblogs.com/lujianwenance/p/5910490.html Block在实际的开发中非常的常用,事件回调.传值.封装成代码块调用等等.很多人都对b ...

随机推荐

  1. PTA数据结构与算法题目集(中文) 7-11

    PTA数据结构与算法题目集(中文)  7-11 7-11 关键活动 (30 分)   假定一个工程项目由一组子任务构成,子任务之间有的可以并行执行,有的必须在完成了其它一些子任务后才能执行.“任务调度 ...

  2. Oracle 数据库创建导入

    Oracle 数据库创建导入 由 Alma 创建, 最后一次修改 2018-06-04 14:37:50 在本章教程中,将教大家如何在Oracle 中创建导入数据库. 注意:本教程中的有些命令您可能并 ...

  3. Centos网络的配置

                                                                                                        ...

  4. android之间的各项信息传输类型

    首先是activity想fragment怎样动态的传输数据: 一:activity与fragment之间进行数据传递是,在Activity中将要传递的数据封装在一Bundle中,使用setArgume ...

  5. 在tap的碎片上与活动进行绑定实现点击事件(日期时间选择以及按钮跳转时间)

    主要是掌握怎样在Fragment类型的.java文件中实现对于文本框或者按钮点击事件的触发操作. 相应的出发时间都是之前的代码.主要是怎样在Fragment怎样实现相应的操作主要是对于getActiv ...

  6. Linux网络安全篇,认识防火墙(一)

    一.概念 防火墙分为软件防火墙和硬件防火墙.我们的主要讨论范围为软件防火墙. 软件防火墙又分为网络型和单一型的管理. 1.单一主机型防火墙 (1)数据包过滤型的Netfilter (2)依据服务软件程 ...

  7. "无用的文本"组件:<s> —— 快应用组件库H-UI

     <import name="s" src="../Common/ui/h-ui/text/c_tag_del"></import> ...

  8. alg-最长公共子串

    class Solution { public: const std::vector<std::string> LongestCommonSubstring(const std::stri ...

  9. docker-compose 基于Dockerfile 安装并启动redis容器的血案

    前言 为了实现"一键部署"的目的,我采用Dockerfile 和 docker-compose来实现自己的目的.这个过程中,我怎么也无法启动自己的redis-server服务. 目 ...

  10. AJ学IOS 之微博项目实战(7)程序启动新特性用UICollectionViewController实现

    AJ分享,必须精品 一:效果 这里实现了大多数app都会有的软件新特性的功能,用的是UICollectionViewController实现的 二:思路 这里用了UICollectionViewCon ...