NSTimer 循环引用分析及解决方案

本文主要是分析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,输出结果如下:

NSTimer_00.jpg

由输入结果可以看到,当返回到控制器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方法,所以计时器将一直处于有效状态。
该情况如下图所示:

NSTimer_01.png

当指向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解决方案

实现原理图如下:
timer_0.jpg

实现代码如下:

#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

其输入结果如下:

NSTimer_02.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容