iOS循环引用深入理解

简介: 循环引用也就是引用之后造成的一个循环用图理解吧

循环引用也就是引用之后造成的一个循环用图理解吧


微信图片_20221018094250.png

image.png


下面是可能造成循环引用的几种情况


1.协议代理


都知道协议的代理对象都用weak修饰,哪为何不用strong,或者copy呢?答案是会造成循环引用。下面就用代码解释吧


#import <Foundation/Foundation.h>
//协议
@protocol TestToolDelegate <NSObject>
- (void)protacolTest:(NSString *)string;
@end
@interface TestTool : NSObject
//使用strong会造成循环引用
@property (nonatomic,strong)id <TestToolDelegate> delegate;
@end


上面是声明的协议,下面是一个类对象,类对象里强引用了delegate对象


下面来看下TestTool的使用


@interface ViewController ()<TestToolDelegate>
@property (nonatomic,strong)TestTool *tool;
@end
@implementation ViewController1
- (void)viewDidLoad {
    [super viewDidLoad];
    self.tool = [[TestTool alloc]init];
//    1协议强引用
    self.tool.delegate = self;
}
- (void)protacolTest:(NSString *)string{
    NSLog(@"%@",string);
}


@property (nonatomic,strong)id <TestToolDelegate> delegate;


如果使用strong修饰则会造成循环引用

所以我们使用weak修饰


@property (nonatomic,weak)id <TestToolDelegate> delegate;


微信图片_20221018094259.png

image.png


2.Block


和协议代理的解释也类似,Block是如何造成循环引用的呢


请看如下代码


#import <Foundation/Foundation.h>
typedef void(^MyBlock)(NSString *string);
@interface TestTool : NSObject
@property (nonatomic,copy) MyBlock block;
- (void)testBlock:(void(^)(NSString *string))block;
@end


#import "TestTool.h"
@implementation TestTool
- (void)testBlock:(void (^)(NSString *))block{
//      如果不将block赋给self.block的话是不会造成循环引用的,可在有的情况下block不在此方法中传值,所以就需要赋值了。
        self.block = block;
//      做完一些事后block返回数据
        dispatch_async(dispatch_get_global_queue(0, 0), ^{
            dispatch_sync(dispatch_get_main_queue(), ^{
                block(@"测试block");
            });
        });
}
@end


@interface ViewController ()<TestToolDelegate>
@property (nonatomic,strong)TestTool *tool;
@end
@implementation ViewController1
- (void)viewDidLoad {
    [super viewDidLoad];
    self.tool = [[TestTool alloc]init];
    [self.tool testBlock:^(NSString *string) {
        NSLog(@"%@",string);
//     如果此时直接使用self的话,就会造成循环引用
        self.view.backgroundColor = [UIColor whiteColor];
    }];
}


正确的使用方法是用weak修饰self;


@interface ViewController ()<TestToolDelegate>
@property (nonatomic,strong)TestTool *tool;
@end
@implementation ViewController1
- (void)viewDidLoad {
    [super viewDidLoad];
    self.tool = [[TestTool alloc]init];
    __weak typeof(self) weakSelf = self;
    [self.tool testBlock:^(NSString *string) {
        NSLog(@"%@",string);
//     如果此时直接使用self的话,就会造成循环引用
        weakSelf.view.backgroundColor = [UIColor whiteColor];
    }];
}


还是用图理解一下吧


微信图片_20221018094302.png

image.png


3.两个类之间的循环引用


一个类是ViewController,一个类是TestTool. ViewController中先强引用了TestTool对象,然后TestTool中也强引用了一个对象ViewController.这样就造成了循环引用。


#import <Foundation/Foundation.h>
@interface TestTool : NSObject
@property (nonatomic,strong) id ID;
@end


@interface ViewController ()
@property (nonatomic,strong)TestTool *tool;
@end
@implementation ViewController1
- (void)viewDidLoad {
    [super viewDidLoad];
    self.tool = [[TestTool alloc]init];
    self.tool.ID = self;
}


图解


微信图片_20221018094306.png

image.png


4.定时器NSTimer


@interface ViewController ()
@property (nonatomic,strong) NSTimer *timer;
@end
@implementation ViewController1
- (void)viewDidLoad {
    [super viewDidLoad];
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:weakSelf selector:@selector(timerClick) userInfo:nil repeats:YES];
}
- (void)timerClick{
    NSLog(@"我在重复");
}
//是否调用dealloc方法,没有则造成了循环引用
- (void)dealloc {
    NSLog(@"ViewController被销毁了");
    [self.timer invalidate];
}


开启定时器以后, timerClick方法会一直执行,即使dismiss此控制器以后,也是一直在打印,而且dealloc方法不会执行.很明显这是循环引用造成了内存泄露,控制器不会被释放.


微信图片_20221018094322.png

image.png


解决办法:由于循环引用的起因是target,则可以包装一个target,让target是另一个对象,而不是ViewController即可,其实NSTimer有一个类方法是Block创建的,亲测不会造成循环引用


- (void)viewDidLoad {
    [super viewDidLoad];
    __weak typeof(self) weakSelf = self;
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {
        weakSelf.view.backgroundColor = [UIColor yellowColor];
    }];
}


微信图片_20221018094326.png

image.png

代码示例


相关文章
|
编解码 API iOS开发
iOS - NSTimer循环引用
iOS - NSTimer循环引用
|
C++ iOS开发
iOS - Block 循环引用
我们知道,循环引用即: 当两个对象A和B, 分别强引用对方,那么就会产生循环引用。即A释放的时候必须先释放B,而B释放的时候必须释放A。导致谁也不能释放 而打破循环引用的方法就是其中一方弱引用另一方
|
编解码 iOS开发
iOS - 循环引用
iOS - 循环引用
|
编解码 iOS开发
|
1月前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
7天前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
91 66
|
17天前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。