iOS 關於NSTimer的迴圈引用

LinXunFeng發表於2019-03-04

現象

在當前控制器(ViewController)的view上新增了一個自定義的view(LXFTimerView),
LXFTimerView在成功建立出來後新增了定時器NSTimer並加入RunLoop開始工作,
當在當前控制器裡將LXFTimerView移除掉後,定時器還在工作,而且LXFTimerView裡的dealloc並沒有呼叫

現象

程式碼

LXFTimerView.m

#import "LXFTimerView.h"

@interface LXFTimerView()
/** 定時器 */
@property(nonatomic, weak) NSTimer *timer;
@end

@implementation LXFTimerView

- (instancetype)initWithFrame:(CGRect)frame {
    if (self = [super initWithFrame:frame]) {
        [self addTimer];
    }
    return self;
}

- (void)dealloc {
    NSLog(@"LXFTimerView - dealloc");
    [self removeTimer];
}

#pragma mark - 定時器方法
/** 新增定時器方法 */
- (void)addTimer {
    // 建立定時器
    if (self.timer) { return; }
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(log) userInfo:nil repeats:YES];
    [[NSRunLoop currentRunLoop] addTimer:self.timer forMode:NSRunLoopCommonModes];
}
/** 移除定時器 */
- (void)removeTimer {
    [self.timer invalidate];
    self.timer = nil;
}
- (void)log {
    NSLog(@"定時器 -- %s", __func__);
}
@end
複製程式碼

ViewController.m

#import "ViewController.h"
#import "LXFTimerView.h"

@interface ViewController ()
/** timerView */
@property(nonatomic, weak) LXFTimerView *timerView;
@end

@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    
    LXFTimerView *timerView = [[LXFTimerView alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 200)];
    timerView.backgroundColor = [UIColor orangeColor];
    self.timerView = timerView;
    [self.view addSubview:timerView];   
}
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    [self.timerView removeFromSuperview];
}
@end
複製程式碼

引用關係

引用關係

問題就出在LXFTimerView與NSTimer之間,在建立定時器時執行

[NSTimer scheduledTimerWithTimeInterval: target: selector: userInfo: repeats:];
複製程式碼

會將LXFTimerView進行強引用,什麼?我怎麼知道?看下圖

NSTimer

翻譯:定時器保持著對target的強引用,直到定時器作廢
那為什麼LXFTimerView中的timer屬性要用weak?? 不用著急,下面即將揭曉~

解決方案

讓定時器指著另一個物件,讓那個物件來執行LXFTimerView中需要執行的方法。
引用關係如下圖所示

LXFWeakTarget

建立一個繼承於NSObject的類 LXFWeakTarget,並提供一個建立定時器的方法(蘋果官方的方法,對scheduledTimerWithTimeInterval進行轉到定義操作【就是command+左鍵】就可以得到)
LXFWeakTarget.h

#import <Foundation/Foundation.h>
@interface LXFWeakTarget : NSObject
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
@end
複製程式碼
#import "LXFWeakTarget.h"

@interface LXFWeakTarget()
@property(nonatomic, weak) id target;
@property(nonatomic, assign) SEL selector;
@end

@implementation LXFWeakTarget
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo {
    // 建立當前類的物件
    LXFWeakTarget *object = [[LXFWeakTarget alloc] init];
    object.target = aTarget;
    object.selector = aSelector;

    return [NSTimer scheduledTimerWithTimeInterval:ti target:object selector:@selector(execute:) userInfo:userInfo repeats:yesOrNo];
}
- (void)execute:(id)obj {
    [self.target performSelector:self.selector withObject:obj]; 
}
@end
複製程式碼

在LXFTimerView.m中匯入LXFWeakTarget的標頭檔案

#import "LXFWeakTarget.h"
複製程式碼

將建立定時器的類改為 LXFWeakTarget

self.timer = [LXFWeakTarget scheduledTimerWithTimeInterval:1.0 target:self selector:@selector(log) userInfo:nil repeats:YES];
複製程式碼

現在再來執行一下程式

執行dealloc

最後縷下思路

  • 我們用一個LXFWeakTarget來替LXFTimerView執行一些操作。
  • 當沒有被定時器強引用的LXFTimerView從父控制元件上被移除時,就會執行dealloc方法,LXFTimerView被銷燬。
  • 將定時器作廢並設為nil,這樣定時器對LXFWeakTarget的引用也沒有了,LXFWeakTarget也會被銷燬。

好,那“為什麼LXFTimerView中的timer屬性要用weak”這個問題就不用多加解析了吧。

相關文章