iOS开发实用技术iOS开发UIKit

多个cell中展示倒计时,本地和服务器时间差异解决方案

2017-09-13  本文已影响935人  Yasin的简书

转载注明出处:http://www.jianshu.com/p/97ec4b8f018c

本文借鉴了IGListKit中多cell通知方案

Demo下载

公司需要做限时抢购的业务,这里面有两个需求点:
1.在多个cell中显示倒计时
在每个cell中添加定时器是不现实的,必定会增加许多性能开销,所以肯定是使用一个定时器,关键在于如何通知到cell刷新UI
2.本地时间可能和服务器时间存在误差
有的手机可能时间没有和网络同步,或者用户故意调整了时间,所以本地时间存在错误的可能,所以就定下使用服务器时间

整体思路

1.在多个cell中显示倒计时

思路是这样的:将需要接收定时器通知的对象注册到定时器单例中,存放在数组里面,当定时器更新的时候遍历数组回调通知

定时器的创建

注意:默认暂停定时器,定时器默认是加载到当前runloop中的,在进行UI界面操作比如滑动列表时,由于在main runloop中NSTimer是同步交付的被“阻塞”,就会导致NSTimer计时出现延误
解决这种延误的方法,一种是在子线程中进行NSTimer的操作,在主线程中修改UI界面显示操作结果;另一种是仍然在主线程中进行NSTimer操作,但是将NSTimer实例加到main runloop的特定mode(模式)中。避免被复杂运算操作或者UI界面刷新所干扰。

timer = Timer.scheduledTimer(withTimeInterval: 1, repeats: true, block: {[unowned self] (_) in
     self.onTimer()
})
//下面这种方法要求onTimer是@objc
//Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(onTimer), userInfo: nil, repeats: true) 
注册通知

这里使用NSHashTable存放注册对象的数组,可以防止循环引用注册对象释放不掉
swift的protocol是一个很好的东西,这样可以更好的规范谁可以注册通知

@objc
protocol TimerListener: class {
    func didOnTimer(announcer: YZTimerUtil, timeInterval: TimeInterval)
}
private let map: NSHashTable<TimerListener> = NSHashTable<TimerListener>.weakObjects()
何时注册删除通知

一开始是在willDisplay、didEndDisplaying方法中进行通知注册的,后来发现没有必要,因为cell创建后就是要接收通知的willDisplay、didEndDisplaying中还要进行cell类型的判断,所以就改为cell- init/deinit中

    deinit {
        YZTimerUtil.sharedInstance.removeListener(listener: self)
    }
  //这里cell使用的是SB
    override func awakeFromNib() {
        super.awakeFromNib()
        YZTimerUtil.sharedInstance.addListener(listener: self)
    }

2.本地时间可能和服务器时间存在误差

这里看项目需求吧,如果项目对时间要求没有那么严格,不做服务器时间对比也行,反正服务器那边会进行判断的,有些对时间要求严格的肯定是要做对比的,比如手机手令的动态码
我的思路是在定时器初始化的时候进行网络请求,拿到服务器的当前时间,然后计算本地和服务器时间的差值,后面就用这个差值进行计算。当然,受网络状态的影响,这个时间可能也不是准确的时间,但是这个时间误差会在一个可控范围内,为了精确时间差,可以每隔一段时间就校准一次,如果要更精准的,可以通过请求的requestTime/responseTime进行算法计算

    /// 从服务器请求最新的时间,简单示例
    func resetServerTime() {
        // 从服务器请求最新的时间
        // 。。。
        var success = true
        
        if success {
            // 请求成功
            serverTimeInterval = 0
        } else {
            // 如果请求失败,隔一段时间再请求一次
            perform(#selector(resetServerTime), with: nil, afterDelay: rloadTimeInterval)
        }
    }

demo里面的代码很详细,也很简单,建议可以看看

上一篇下一篇

猜你喜欢

热点阅读