美文网首页
面试题:NSTimer 循环引用分析及解决方案

面试题:NSTimer 循环引用分析及解决方案

作者: 常在士心 | 来源:发表于2021-09-14 15:30 被阅读0次

    <article class="_2rhmJa">

    本文主要是分析NSTimer 循环引用的原因及解决方案:

    1. NSTimer循环引用的原因;
    2. 苹果API接口解决方案;(iOS 10.0)
    3. NSProxy解决方案;
    4. Block解决方案;

    一.NSTimer循环引用的案例:

    1.对定时器SJTimer进行简单封装

    //SJTimer.h文件
    #import <Foundation/Foundation.h>
    @interface SJTimer : NSObject
    //开启定时器
    - (void)startTimer;
    //暂停定时器
    - (void)stopTimer;
    @end
    
    //SJTimer.m文件
    #import "SJTimer.h"
    @implementation SJTimer
    {
        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",NSStringFromClass([SJTimer class]),__func__);
        [_timer invalidate];
    }
    @end
    
    

    2.创建两个控制器A,B;由控制器A跳转到控制器B;在控制器B中创建一个定时器timer,点击开始按钮,开启定时器;点击返回按钮,则返回控制器A;

    //控制器A的.m文件
    #import "ViewController.h"
    #import "SJSecondVC.h"
    @interface ViewController ()
    @end
    @implementation ViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
    
    }
    
     //跳转到控制器B
    - (IBAction)jump:(UIButton *)sender {
    
     SJSecondVC *secondVC = [[SJSecondVC alloc] init];
    [self presentViewController:secondVC animated:YES completion:^{
    
        }];
    }
    @end
    
    
    //控制器B的.m文件
    #import "SJSecondVC.h"
    #import "SJTimer.h"
    
    @interface SJSecondVC ()
    @property (nonatomic, strong) SJTimer *timer;
    @end
    
    @implementation SJSecondVC
    
    //开启定时器
    - (IBAction)start:(id)sender {
        SJTimer * timer = [[SJTimer alloc] init];
        self.timer = timer;
        [timer startTimer]; 
    }
    
    //返回控制器A
    - (IBAction)back:(UIButton *)sender {
    
        [self dismissViewControllerAnimated:YES completion:^{     
        }];
    }
    
    //控制器B销毁时,会自动调用该方法
    - (void)dealloc{
    
        NSLog(@"%@-----%s",NSStringFromClass([SJSecondVC class]),__func__);
    
    }
    - (void)viewDidLoad {
        [super viewDidLoad];
        // Do any additional setup after loading the view from its nib.
    }
    @end
    
    

    3.运行程序,由控制器A跳转到控制器B,并开启定时器,然后返回到控制A,输出结果如下:

    image

    由输入结果可以看到,当返回到控制器A后,控制器B已经被销毁,但SJTimer的实例对象没有被销毁,计时器仍然在执行任务。这是什么原因呢?

    二.NSTimer循环引用分析

    下面的方法可以创建计时器,并将其预先安排到当前运行循环(Run Loop)当中:

    + (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
    
    

    参数target和selector表示计时器将在哪个对象上调用哪个方法,repeats表示是否重复执行任务。
    计时器会保留其目标对象,等到自身“失效”时再释放此对象。
    (1)当repeats设置为NO时,执行完相关任务之后,计时器会自动失效;
    (2)当调用invalidate方法时,可以令计时器失效;
    因此将计时器设置成重复模式时,很容易导致“循环引用”的问题,必须自己调用invalidate方法,才能停止计时器。

    在上面的案例中,当我们在控制器B中创建SJTimer类的实例对象timer,并调用其startTimer方法时,由于NSTimer的目标对象是self,所以NSTimer要保留该实例timer。然而,因为计时器是用实例变量存放的,所以实例对象timer也保留了计时器。因此产生了“保留环”。

    如果能在某一刻打破该保留环,则程序不会出问题。若要打破保留环,只能改变实例变量或令计时器无效。所以当调用stopTimer方法,或者令系统将实例对象timer回收时才能打破保留环。

    但是在团队开发中,我们无法保证stopTimer一定会被调用,而且这种做法也不是一种很好的解决方案。另外,如果想在系统回收本类实例的过程中令计时器无效,从而打破保留环,又会陷入死结。因为在计时器对象有效时,SJTimer实例的自动计数器绝不会为0,因此系统也绝不会将其回收。此时,又没有调用invalidate方法,所以计时器将一直处于有效状态。
    该情况如下图所示:

    image

    当指向SJTimer实例的最后一个外部引用被移走之后,该实例仍然继续存活。因为计时器还保留着它。而计时器对象也不可能被系统释放,因为实例中还有一个强引用正在指向它。于是,导致循环引用,内存就泄漏了。这种内存泄露问题尤为重要,因为计时器还将继续反复的执行轮训任务。倘若每次轮训时都要联网下载数据的话,那么程序会一直下载数据,这又更容易导致其他内存泄漏问题了。
    NSTimer循环引用的原因到此分析完毕。下面来看看NSTimer循环引用的解决方案。

    三.苹果API接口解决方案(iOS 10.0以上)

    在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方法;

    四.NSProxy解决方案

    实现原理图如下:

    image

    实现代码如下:

    #import <Foundation/Foundation.h>
    
    NS_ASSUME_NONNULL_BEGIN
    
    @interface MyProxy : NSProxy
    
    - (instancetype)initWithObjc:(id)objc;
    + (instancetype)proxyWithObjc:(id)objc;
    
    @end
    
    NS_ASSUME_NONNULL_END
    
    
    
    #import "MyProxy.h"
    
    @interface MyProxy()
    
    @property(nonatomic,weak) id objc;
    
    @end
    
    @implementation MyProxy
    
    - (instancetype)initWithObjc:(id)objc{
        self.objc = objc;
        return self;
    }
    
    + (instancetype)proxyWithObjc:(id)objc{
    
        return [[self alloc] initWithObjc:objc];
    }
    
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    
        return [self.objc methodSignatureForSelector:aSelector];
    
    }
    - (void)forwardInvocation:(NSInvocation *)invocation {
    
        if ([self.objc respondsToSelector:invocation.selector]) {
    
            [invocation invokeWithTarget:self.objc];
        }
    }
    
    @end
    
    
    
    - (void)viewDidLoad {
        [super viewDidLoad];
        // Do any additional setup after loading the view from its nib.
    
        _count = 0;
    
        MyProxy *proxy = [[MyProxy alloc] initWithObjc:self];
        _timer = [NSTimer timerWithTimeInterval:1 target:proxy selector:@selector(test_000) userInfo:nil repeats:YES];
    
        [[NSRunLoop currentRunLoop] addTimer:_timer forMode:NSRunLoopCommonModes];
    
    }
    
    - (void)test_000{
    
        NSLog(@"------%d",_count++);
    }
    
    -(void)dealloc{
        NSLog(@"---dealloc----");
        [_timer invalidate];
    }
    
    

    五.Block解决方案

    从计时器本身入手,很难解决该问题,可以要求外界对象在释放最后一个指向本实例的引用之前,必须调用stopTimer方法。然而这种情况无法通过代码检测出来。此外,在团队开发中,我们无法保证其他开发人员一定会调用此方法。我们可以通过“Block”来解决该问题。
    其代码如下:

    //NSTimer+SJSafeTimer.h文件
    
    #import <Foundation/Foundation.h>
    @interface NSTimer (SJSafeTimer)
    
    + (NSTimer *)SJ_ScheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void(^)(void))block;
    
    @end
    
    //NSTimer+SJSafeTimer.m 文件
    #import "NSTimer+SJSafeTimer.h"
    
    @implementation NSTimer (SJSafeTimer)
    
    + (NSTimer *)SJ_ScheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void(^)(void))block{
    
        return [self scheduledTimerWithTimeInterval:interval target:self selector:@selector(handler:) userInfo:[block copy] repeats:repeats];
    }
    
    + (void)handler:(NSTimer *)timer{
    
        void (^block)(void) = timer.userInfo;
        if (block) {
            block();
        }
    }
    @end
    
    

    该方案是将计时器所应执行的任务封装成"Block",在调用计时器函数时,把block作为userInfo参数传进去。userInfo参数用来存放"不透明值",只要计时器有效,就会一直保留它。在传入参数时要通过copy方法,将block拷贝到"堆区",否则等到稍后要执行它的时候,该blcok可能已经无效了。计时器现在的target是NSTimer类对象,这是个单例,因此计时器是否会保留它,其实都无所谓。此处依然有保留环,然而因为类对象(class object)无需回收,所以不用担心。
    该方案本身不能解决问题,它只是提供了解决问题所需的工具。现在我们将使用新分类中的方法来创建计时器,将SJTimer中的方法startTimer修改如下:

    - (void)startTimer{
    
        _timer = [NSTimer SJ_ScheduledTimerWithTimeInterval:1.0 repeats:YES block:^{
    
            [self  work];
    
        }];
    }
    
    

    这段代码,还是会有保留环。因为block捕获了self变量,所以block要保留实例。而计时器又通过userInfo参数保留了block。最后,实例对象本身还有保留计时器。我们要打破保留环,只需改用weak引用即可:

    - (void)startTimer{
        __weak SJTimer *weakSelf = self;
        _timer = [NSTimer SJ_ScheduledTimerWithTimeInterval:1.0 repeats:YES block:^{
    
            __strong SJTimer *strongSelf = weakSelf;
            [strongSelf work];
    
        }];
    }
    
    

    这里,我们先定义了一个弱引用,令其指向self,然后使block捕获这个弱引用,而不是直接捕获普通的self变量(即self不会被计时器所保留)。当block开始执行时,立刻生成strong引用,以保证实例对象在执行期间持续存活。
    当外界指向SJTimer实例对象的最后一个引用将其释放,则该实例就会被系统回收。回收过程中还会调用计时器的invalidate方法,这样计时器就不会再继续执行任务了。

    最后我们在控制器B中调用:

    
    @interface SJSecondVC ()
    
    @end
    
    @implementation SJSecondVC
    {
        SJTimer *_timer;
    }
    
    //开启定时器
    - (IBAction)start:(id)sender {
    
        _timer = [[SJTimer alloc] init];
        [_timer startPolling];
    
    }
    
    //返回控制器a
    - (IBAction)back:(UIButton *)sender {
    
        [self dismissViewControllerAnimated:YES completion:^{
    
        }];
    }
    
    - (void)dealloc{
    
        NSLog(@"%@-----%s",NSStringFromClass([SJSecondVC class]),__func__);
    
    }
    - (void)viewDidLoad {
        [super viewDidLoad];
        // Do any additional setup after loading the view from its nib.
    
    }
    
    - (void)didReceiveMemoryWarning {
        [super didReceiveMemoryWarning];
        // Dispose of any resources that can be recreated.
    }
    
    @end
    
    

    其输入结果如下:

    image

    </article>

    16人点赞

    iOS文集

    作者:WSJay
    链接:https://www.jianshu.com/p/33d8931e60ee
    来源:简书

    相关文章

      网友评论

          本文标题:面试题:NSTimer 循环引用分析及解决方案

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