美文网首页iOS深入原理
iOS之NSTimer循环引用的解决方案

iOS之NSTimer循环引用的解决方案

作者: Amor瑾年v | 来源:发表于2021-03-09 11:02 被阅读0次

    前言

    在使用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 <Foundation/Foundation.h>
    @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 <Foundation/Foundation.h>
    
    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来解决循环引用

    原理如下图:

    image

    NSProxy解决循环引用原理.png

    代码如下:

    //PFProxy.h
    #import <Foundation/Foundation.h>
    
    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造成循环。

    原文:https://juejin.cn/post/6844903968250789896

    相关文章

      网友评论

        本文标题:iOS之NSTimer循环引用的解决方案

        本文链接:https://www.haomeiwen.com/subject/bbxvqltx.html