Block 的循环引用:如何产生 和 解决办法


Block 的循环引用


//  定义 block 的时候,会对外部变量做一次 copy,  强引用


//  ViewController.m
#import "ViewController.h"
#import "NetworkTools.h"
@interface ViewController ()
@property (nonatomic, strong) NetworkTools *tools;
@end
@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
     self.tools = [[NetworkTools alloc] init];
     //
    [self.tools loadData:^(NSString *html) {
       
         NSLog(@"%@ %@", html, self.view);
    }];
}
- (void)dealloc {
    NSLog(@"控制器 88");
}
@end
//  NetworkTools.h
#import <Foundation/Foundation.h>
@interface NetworkTools : NSObject
- (void)loadData:(void (^)(NSString *html))finished;
@end
//  NetworkTools.m
#import "NetworkTools.h"

@interface NetworkTools()
@property (nonatomic, copy) void (^finishedBlock)(NSString *);
@end

@implementation NetworkTools
/**
 block 是一组提前准备好的代码,在需要的时候执行
 可以当作参数传递
 
 在异步的方法中,如果能够执行 block,就直接执行!
 如果不能直接执行 block,通常需要定义一个属性,记录 block,在需要的时候执行
 */
 
// 间接调用
- (void)loadData:(void (^)(NSString *))finished {
    
    // 1. 使用属性记录 block
    self.finishedBlock = finished;
    
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        
        NSLog(@"异步处理");
        
        // 模拟延时
        [NSThread sleepForTimeInterval:2.0];
        
        dispatch_async(dispatch_get_main_queue(), ^{
            NSLog(@"准备回调");
            
            // 2. 由 working 调用 finished
            [self working];
        });
    });
}


- (void)working {
    // 执行回调函数
    if (self.finishedBlock != nil) {
        self.finishedBlock(@"hello .....");
    }
}


- (void)dealloc {
    NSLog(@"network tools 88");
}
@end

011502_ytaM_2432308.png

如何产生循环引用

011853_romP_2432308.png


解决办法

#import "ViewController.h"
#import "NetworkTools.h"
@interface ViewController ()
@property (nonatomic, strong) NetworkTools *tools;
@end
@implementation ViewController
// 1. 解除循环引用,需要注意打断引用链条即可!
- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 局部变量不会产生循环应用,全局属性会产生循环引用
    self.tools = [[NetworkTools alloc] init];
    
    // 1. 定义 block 的时候,会对外部变量做一次 copy,会对 self 进行强引用
    
    // 解除循环引用方法1
    // __weak 是 iOS 5.0 推出的
    // 如果异步操作没有完成,释放控制器,__weak 本身是弱引用
    // 当异步执行完毕,进行回调,self 已经被释放,无法访问属性,也无法调用方法
    // __weak 相当于 weak,不会做强引用,但是如果对象被释放,执行的地址,会指向 nil
    // __weak 更安全
    __weak typeof(self) weakSelf = self;
    
    // 解除循环引用方法2
    // __unsafe_unretained 是 iOS 4.0 推出的
    // MRC 经典错误,EXC_BAD_ACCESS 坏内存访问,野指针
    // 相当于 assign,不会做强引用,但是如果对象被释放,内存地址保持不变,如果此时再调用,就会出现野指针访问
//    __unsafe_unretained typeof(self) weakSelf = self;
    
    [self.tools loadData:^(NSString *html) {
        
        // 工作中会看到一个代码,流传出来的,一直没有人太过深究,记住了套路
        // strongSelf 强引用,对 weakSelf 进行强引用,本意,希望在异步完成后,继续执行回调代码
        //然而并没有什么作用!!!!!!!!
        __strong typeof(self) strongSelf = weakSelf;
        
        NSLog(@"%@ %@", html, strongSelf.view);
    }];
}

- (void)dealloc {
    NSLog(@"控制器 88");
}
@end






转载于:https://my.oschina.net/TaciturnKnightYQ/blog/539138

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值