您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關iOS中NSTimer循環引用的示例分析,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
在當前控制器(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進行強引用,什么?我怎么知道?看下圖
翻譯:定時器保持著對target的強引用,直到定時器作廢 那為什么LXFTimerView中的timer屬性要用weak?? 不用著急,下面即將揭曉~
解決方案
讓定時器指著另一個對象,讓那個對象來執行LXFTimerView中需要執行的方法。 引用關系如下圖所示
創建一個繼承于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];
現在再來執行一下程序
最后縷下思路
我們用一個LXFWeakTarget來替LXFTimerView執行一些操作。
當沒有被定時器強引用的LXFTimerView從父控件上被移除時,就會執行dealloc方法,LXFTimerView被銷毀。
將定時器作廢并設為nil,這樣定時器對LXFWeakTarget的引用也沒有了,LXFWeakTarget也會被銷毀。
好,那“為什么LXFTimerView中的timer屬性要用weak”這個問題就不用多加解析了吧。
關于“iOS中NSTimer循環引用的示例分析”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。