nstimer循环引用_iOS之NSTimer循环引用的解决方案

前言

在使用NSTimer,如果使用不得当特别会引起循环引用,造成内存泄露。所以怎么避免循环引用问题,下面我提出几种解决NSTimer的几种循环引用。

原因

当你在ViewController(简称VC)中使用timer属性,由于VC强引用timer,timer的target又是VC造成循环引用。当你在VC的dealloc方法中销毁timer,

发现VC被pop,VC的dealloc方法没走,VC在等timer释放才走dealloc,timer释放在dealloc中,所以引起循环引用。

解决方案在ViewController执行dealloc前释放timer(不推荐)

对定时器NSTimer封装

苹果API接口解决方案(iOS 10.0以上可用)

使用block进行解决

使用NSProxy进行解决

一、在ViewController执行dealloc前释放timer(不推荐)可以在viewWillAppear中创建timer

可以在viewWillDisappear中销毁timer

二、对定时器NSTimer封装到PFTimer中

代码如下://PFTimer.h文件

#import

@interface PFTimer : NSObject

//开启定时器

- (void)startTimer;

//暂停定时器

- (void)stopTimer;

@end

复制代码

在PFTimer.m文件中代码如下:#import "PFTimer.h"

@implementation PFTimer {

NSTimer *_timer;

}

- (void)stopTimer{

if (_timer == nil) {

return;

}

[_timer invalidate];

_timer = nil;

}

- (void)startTimer{

_timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(work) userInfo:nil repeats:YES];

}

- (void)work{

NSLog(@"正在计时中。。。。。。");

}

- (void)dealloc{

NSLog(@"%s",__func__);

[_timer invalidate];

_timer = nil;

}

@end

复制代码

在ViewController中使用代码如下:#import "ViewController1.h"

#import "PFTimer.h"

@interface ViewController1 ()

@property (nonatomic, strong) PFTimer *timer;

@end

@implementation ViewController1

- (void)viewWillDisappear:(BOOL)animated {

[super viewWillDisappear:animated];

}

- (void)viewDidLoad {

[super viewDidLoad];

self.title = @"VC1";

self.view.backgroundColor = [UIColor whiteColor];

//自定义timer

PFTimer *timer = [[PFTimer alloc] init];

self.timer = timer;

[timer startTimer];

}

- (void)dealloc {

[self.timer stopTimer];

NSLog(@"%s",__func__);

}

复制代码

运行打印结果:-[ViewController1 dealloc]

-[PFTimer dealloc]

复制代码

这个方式主要就是让PFTimer强引用NSTimer,NSTimer强引用PFTimer,避免让NSTimer强引用ViewController,这样就不会引起循环引用,然后在dealloc方法中执行NSTimer的销毁,相对的PFTimer也会进行销毁了。

三、苹果系统API可以解决(iOS10以上)

在iOS 10.0以后,苹果官方新增了关于NSTimer的三个API:+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)interval repeats:

(BOOL)repeats block:(void (^)(NSTimer *timer))block

API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));

+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:

(BOOL)repeats block:(void (^)(NSTimer *timer))block

API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));

- (instancetype)initWithFireDate:(NSDate *)date interval:

(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block

API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));

复制代码

这三个方法都有一个Block的回调方法。关于block参数,官方文档有说明:the timer itself is passed as the parameter to this block when executed

to aid in avoiding cyclical references。

复制代码

翻译过来就是说,定时器在执行时,将自身作为参数传递给block,来帮助避免循环引用。使用很简单,但是要注意两点:1.避免block的循环引用,使用__weak和__strong来避免2.在持用NSTimer对象的类的方法中-(void)dealloc调用NSTimer 的- (void)invalidate方法;

四、使用block来解决

通过创建一个NSTimer的category名字为PFSafeTimer,在NSTimer+PFSafeTimer.h代码如下:#import

NS_ASSUME_NONNULL_BEGIN

@interface NSTimer (PFSafeTimer)

+ (NSTimer *)PF_ScheduledTimerWithTimeInterval:(NSTimeInterval)timeInterval block:

(void(^)(void))block repeats:(BOOL)repeats;

@end

NS_ASSUME_NONNULL_END

复制代码

在NSTimer+PFSafeTimer.m中的代码如下:#import "NSTimer+PFSafeTimer.h"

@implementation NSTimer (PFSafeTimer)

+ (NSTimer *)PF_ScheduledTimerWithTimeInterval:(NSTimeInterval)timeInterval block:(void(^)(void))block repeats:(BOOL)repeats {

return [NSTimer scheduledTimerWithTimeInterval:timeInterval target:self selector:@selector(handle:) userInfo:[block copy] repeats:repeats];

}

+ (void)handle:(NSTimer *)timer {

void(^block)(void) = timer.userInfo;

if (block) {

block();

}

}

@end

复制代码

该方案主要要点:将计时器所应执行的任务封装成"Block",在调用计时器函数时,把block作为userInfo参数传进去。

userInfo参数用来存放"不透明值",只要计时器有效,就会一直保留它。

在传入参数时要通过copy方法,将block拷贝到"堆区",否则等到稍后要执行它的时候,该blcok可能已经无效了。

计时器现在的target是NSTimer类对象,这是个单例,因此计时器是否会保留它,其实都无所谓。此处依然有保留环,然而因为类对象(class object)无需回收,所以不用担心。

再调用如下:#import "ViewController1.h"

#import "PFTimer.h"

#import "NSTimer+PFSafeTimer.h"

@interface ViewController1 ()

//使用category

@property (nonatomic, strong) NSTimer *timer1;

@end

@implementation ViewController1

- (void)viewWillDisappear:(BOOL)animated {

[super viewWillDisappear:animated];

}

- (void)viewDidLoad {

[super viewDidLoad];

self.title = @"VC1";

self.view.backgroundColor = [UIColor whiteColor];

__weak typeof(self) weakSelf = self;

self.timer1 = [NSTimer PF_ScheduledTimerWithTimeInterval:1.0 block:^{

__strong typeof(self) strongSelf = weakSelf;

[strongSelf timerHandle];

} repeats:YES];

}

//定时触发的事件

- (void)timerHandle {

NSLog(@"正在计时中。。。。。。");

}

- (void)dealloc {

// [self.timer stopTimer];

NSLog(@"%s",__func__);

}

复制代码

如果在block里面直接调用self,还是会保留环的。因为block对self强引用,self对timer强引用,timer又通过userInfo参数保留block(强引用block),这样就构成一个环block->self->timer->userinfo->block,所以要打破这个环的话要在block里面弱引用self。

使用NSProxy来解决循环引用

原理如下图:

NSProxy解决循环引用原理.png

代码如下://PFProxy.h

#import

NS_ASSUME_NONNULL_BEGIN

@interface PFProxy : NSProxy

//通过创建对象

- (instancetype)initWithObjc:(id)object;

//通过类方法创建创建

+ (instancetype)proxyWithObjc:(id)object;

@end

NS_ASSUME_NONNULL_END

复制代码

在PFProxy.m文件中写代码#import "PFProxy.h"

@interface PFProxy()

@property (nonatomic, weak) id object;

@end

@implementation PFProxy

- (instancetype)initWithObjc:(id)object {

self.object = object;

return self;

}

+ (instancetype)proxyWithObjc:(id)object {

return [[self alloc] initWithObjc:object];

}

- (void)forwardInvocation:(NSInvocation *)invocation {

if ([self.object respondsToSelector:invocation.selector]) {

[invocation invokeWithTarget:self.object];

}

}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {

return [self.object methodSignatureForSelector:sel];

}

@end

复制代码

在使用的时候如下代码:#import "ViewController1.h"

#import "PFProxy.h"

@interface ViewController1 ()

//使用NSProxy

@property (nonatomic, strong) NSTimer *timer2;

@end

@implementation ViewController1

- (void)viewWillDisappear:(BOOL)animated {

[super viewWillDisappear:animated];

}

- (void)viewDidLoad {

[super viewDidLoad];

self.title = @"VC1";

self.view.backgroundColor = [UIColor whiteColor];

PFProxy *proxy = [[PFProxy alloc] initWithObjc:self];

self.timer2 = [NSTimer scheduledTimerWithTimeInterval:1.0 target:proxy selector:@selector(timerHandle) userInfo:nil repeats:YES];

}

//定时触发的事件

- (void)timerHandle {

NSLog(@"正在计时中。。。。。。");

}

- (void)dealloc {

[self.timer2 invalidate];

self.timer2 = nil;

NSLog(@"%s",__func__);

}

@end

复制代码

当pop当前viewController时候,打印结果:-[ViewController1 dealloc]

复制代码

通过PFProxy这个伪基类(相当于ViewController1的复制类),避免直接让timer和viewController造成循环。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值