Object-C定时器,封装GCD定时器的必要性!!! (一)

实际项目开发中经常会遇到延迟某件任务的执行,或者让某件任务周期性的执行。然后也会在某些时候需要取消掉之前延迟执行的任务。

iOS中延迟操作有三种解决方案:

1、NSObject的方法:(对象方法)

- (void)performSelector:(SEL)aSelector withObject:(nullable id)anArgument afterDelay:(NSTimeInterval)delay;

 

2、使用NSTimer的方法:

 

+ (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)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;

//创建后会默认添加到NSDefaultRunLoopMode中,这个方法创建的定时器不会自动销毁,需要手动销毁,会被self强引用着,不特殊处理就会产生强引用循环,造成内存泄露. 一定不要使用这个方法,请使用下面的方法替代这个方法.

------------------------------------------------------------------------------

+ (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));

//创建后会默认添加到NSDefaultRunLoopMode中,添加到block中,系统回自己处理(不会强引用),系统会调用dealloc方法我们在此处销毁timer即可

------------------------------------------------------------------------------

 

 

模式:以下两种模式同一时刻只能是一种模式

NSDefaultRunLoopMode(默认模式)

UITrackingRunLoopMode(如果控制器的view上面有滚动视图,但手指拖拽滚动视图的时候,就会进入该模式.一般不会将定时器加入到这个模式中,如果想在滚动视图的时候,定时器同样起效一般会加入到下面的模式) 

---------------------------------------------------------------------------------------

NSRunLoopCommonModes:上面两个模式都能运行

 

 

3、使用GCD的方法:

dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{

     //延迟执行的代码 

  

});

 

一般情况下,我们选择使用GCD的dispatch_after。

因为如果不用GCD,需要注意以下三个细节:

1.必须保证有一个活跃的runloop。

当一个应用启动时,系统会开启一个主线程,并且把主线程跑起来,并且主线程的runloop是不会停止的。所以,当这两个方法在主线程可以被正常调用。但实际编程中大部分逻辑处理是放在子线程中执行的。而子线程的runloop是默认关闭的。如果不手动激活runloop,performSelector和scheduledTimerWithTimeInterval的调用将是无效的。

2.NSTimer、performSelector的创建与撤销必须在同一个线程操作。

3.内存有潜在泄露的风险 

4.NSTimer相对于Dispatch定时器来说不准时.

+ (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));

//当然使用这个方法,不会产生强引用循环(系统已经帮我们处理了),我们只需要在对应的dealloc方法中销毁定时器就OK了

 

但是dispatch_after有个致命的弱点:dispatch_after一旦执行后,就不能撤销了。

 

其实GCD也有timer的功能。

 

// 1.获得队列

 dispatch_queue_t queue = dispatch_get_global_queue(0, 0);

  //dispatch_queue_t queue = dispatch_get_main_queue();

 

 // 2.创建一个定时器(dispatch_source_t本质还是个OC对象

 self.timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);

 

 // 3.设置timer执行的事件

dispatch_time_t start = dispatch_time(DISPATCH_TIME_NOW,1.0 * NSEC_PER_SEC);//1.0秒之后开始执行

 uint64_t interval = (uint64_t)(2.0 * NSEC_PER_SEC);//每隔2.0秒执行一次

 dispatch_source_set_timer(self.timer, start, interval, 0);

 

 //4. 设置回调

 dispatch_source_set_event_handler(self.timer, ^{

      // 取消timer 或者做其他事情

      dispatch_cancel(self.timer);

      self.timer = nil;

 });

 

 //5.启动定时器/激活timer

 dispatch_resume(self.timer);

 

这样我们就规避了NSTimer的三个缺陷。

我靠... 这也太复杂了!!! 而且还没有repeats选项

我们能不能像NSTimer那样使用呢?答案:当然有了!!!

没错! 我们将重复的代码封装起来,开放几个供外界调用的参数!

有了思路写代码就很简单了!

 

 

更多内容--> 博客导航 每周一篇哟!!!

 

 

有任何关于iOS开发的问题!欢迎下方留言!!!或者邮件lieryangios@126.com 虽然我不一定能够解答出来,但是我会请教iOS开发高手!!!解答您的问题!!!

 

 

 

详细设计请看下一篇: Object-C定时器,封装GCD定时器的必要性!!! (二)

 

 

posted on 2017-04-01 09:35  人生为代码而活  阅读(2039)  评论(0编辑  收藏  举报

导航