有两种方案解决定时器前后台切换,倒计时停止显示不准确的问题。
1.准确及时方案:创建NSTimer定时器,监控前后台通知和全局变量记录时间。定时器开启的页面或组件监控进入后台通知,收到通知记录当前时间,当接收到进入前台通知时计算当前时间和记录的时间差,用倒计时的时间减去这个时间差就时剩下的时间,立即显示这个倒计时时间。这样及时准确。不能解决用户修改系统时间这种非常规操作。当然可以通过监控应用进入后台和服务器时间对时,算出这个时间差,再进行修正。这个完美方案需要后台配合,仅仅依赖app时搞不定所有的。
2.切换到后台存在延迟刷新的问题解决方案:起线程记录时间。
-(void)startTimer{ int seconds = 1000; dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0); dispatch_source_t timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue); dispatch_source_set_timer(timer, DISPATCH_TIME_NOW, 100.0 * NSEC_PER_SEC, 0 * NSEC_PER_SEC); NSDate *endTime = [NSDate dateWithTimeIntervalSinceNow:seconds]; dispatch_source_set_event_handler(timer, ^{ int interval = [endTime timeIntervalSinceNow]; if (interval <= 0) { dispatch_source_cancel(timer); dispatch_async(dispatch_get_main_queue(), ^{ NSLog(@"0"); }); } else { dispatch_async(dispatch_get_main_queue(), ^{ NSLog(@"%d", interval); }); } }); dispatch_resume(timer); }
NSDate *endTime = [NSDate dateWithTimeIntervalSinceNow:seconds];
是记录定时器开启时的时间加上定时的时间(终止时间)。
int interval = [endTime timeIntervalSinceNow];
是计算终止时间和现在的时间差,单位是:秒。
这个是定时间隔到了更新定时间隔。
当然这种方案存在一个显著的问题。当应用切换到后台,然后切换到前台,这个定时器回调而是在0到1个定时间隔之内。若这个定时间隔很大,你在应用切换到前台时显示的是还是以前的时间,并且这个错误时间被更新需要比较长的时间。这个是它存在不及时的问题。
当然它的优点也和显然:代码和简单,不需要接受通知和生命全局变量记录时间,当定时间隔很小时,这种延迟很小,对即时性要求不太高的场景建议采用这种方式。
当然它也不能单独解决用户修改系统时间的问题。