NSTimer解决循环引用

问题

在使用NSTimer的时候,我们会遇到按理说控制器会调用dealloc的情况下并没有调用,这就是因为在初始化NSTimer的时候,传入的target会被NSTimer强引用,并且控制器强引用NSTimer,所以产生循环引用。

使用如下代码,就可以看到在TwoViewController退出的时候,dealloc并没有调用

@interface TwoViewController ()
@property (nonatomic, strong) NSTimer *timer;
@end

@implementation TwoViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.timer = [NSTimer timerWithTimeInterval:1.0 target:self selector:@selector(timerAction) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
}

- (void)timerAction
{
    NSLog(@"%s", __func__);
}

- (void)dealloc
{
    [self.timer invalidate];
    NSLog(@"%s", __func__);
}
循环引用产生的原因

上面看到的图片都是靠项目运行期间产生的问题推测出来的,有什么方法可以判断猜测的正确性呢?下面我会介绍源代码方式和解决循环引用的方式。

源代码验证循环引用

因为iOS Foundation框架是闭源的,所以并没有直接的代码供用户查看源码。但是我们可以通过 GNUstep 开源项目进行查看,它将Cocoa的OC库重新开源实现了一遍,因此对我们软件开发具有一定的参考价值。

+ (NSTimer*) timerWithTimeInterval: (NSTimeInterval)ti
                target: (id)object
              selector: (SEL)selector
              userInfo: (id)info
               repeats: (BOOL)f
{
  return AUTORELEASE([[self alloc] initWithFireDate: nil
                       interval: ti
                         target: object
                       selector: selector
                       userInfo: info
                        repeats: f]);
}
.
.
.
- (id) initWithFireDate: (NSDate*)fd
           interval: (NSTimeInterval)ti
         target: (id)object
           selector: (SEL)selector
           userInfo: (id)info
        repeats: (BOOL)f
{
  if (ti <= 0.0)
    {
      ti = 0.0001;
    }
  if (fd == nil)
    {
      _date = [[NSDate_class allocWithZone: NSDefaultMallocZone()]
        initWithTimeIntervalSinceNow: ti];
    }
  else
    {
      _date = [fd copyWithZone: NSDefaultMallocZone()];
    }
  _target = RETAIN(object);
  _selector = selector;
.
.
.
@interface NSTimer : NSObject
{
#if GS_EXPOSE(NSTimer)
@public
  NSDate    *_date;     /* Must be first - for NSRunLoop optimisation */
  BOOL      _invalidated;   /* Must be 2nd - for NSRunLoop optimisation */
  BOOL      _repeats;
  NSTimeInterval _interval;
  id        _target;
  SEL       _selector;
  id        _info;

查看上述三个代码片段,通过+ timerWithTimeInterval:target:selector:userInfo:repeats:定位到_target可以看到,项目是通过强引用,引用这个_target。因此,产生循环引用也就不奇怪了。

代码解决方法

首先我们可以先看一下如下图,我们可以添加一个中间类,将TimerProxytarget设为弱引用并指向当前控制器就不会产生循环引用了

解决逻辑图

代码实现如下:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.timer = [NSTimer timerWithTimeInterval:1.0 target:[TimerProxy timerProxyWithTarget:self] selector:@selector(timerAction) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];

}

@interface TimerProxy : NSObject
@property (nonatomic, weak) id target;
+ (instancetype)timerProxyWithTarget:(id)target;
@end

@implementation TimerProxy
+ (instancetype)timerProxyWithTarget:(id)target
{
    TimerProxy *instance = [TimerProxy new];
    instance.target = target;
    
    return instance;
}

- (id)forwardingTargetForSelector:(SEL)aSelector
{
    return self.target;
}
@end

这里使用到了runtime消息转发机制,将当前原本发送到target(TimerProxy类)selector转发到当前控制器了,避免方法找不到错误,这样就解决循环引用。

当然了,也可以使用YYkit那套分类NSTimer+YYAdd,他将target指向了NSTimer类对象,并且通过block传递selector,iOS10之后,也提供了类似YYkit的做法。他们相同的做法就是避免target指向当前view或者控制器。

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