app倒计时处理,以及客户端和服务器之间的时间同步

最近项目中用到了很多倒计时的,比如获取验证码的页面、直播倒计时页面;
这样就会导致app里有多个NSTimer在倒计时,这样对性能有所损耗;
所以考虑加个TimerService类,专门做倒计时处理
TimerService内部负责维护唯一一个定时器,同时提供添加和移除监听者的接口以及监听者需要实现的协议protocol,内部通过HashTable来存储监听者,每次定时器回调,遍历所有监听者进行回调,监听者在不需要接收定时器回调的时候只需要从TimerService中移除即可

  • (void)addListener:(id<TimerListener>)listener {
    TIMER_SERVICE_LOCK(self.operationsLock)
    if(![self.map containsObject:listener]){
    [self.map addObject:listener];
    if(self.map.count > 0){
    //启动
    [self startTimer];
    }
    }
    TIMER_SERVICE_UNLOCK(self.operationsLock)
    }
  • (void)removeListener:(id<TimerListener>)listener {
    TIMER_SERVICE_LOCK(self.operationsLock)
    if([self.map containsObject:listener]){
    [self.map removeObject:listener];
    if(self.map.count == 0){
    //暂停
    [self stopTimer];
    }
    }
    TIMER_SERVICE_UNLOCK(self.operationsLock)
    }

直播倒计时存在客户端和服务器之间的时间同步
提供方法不管用户是否调整手机时间获取到的时间不受影响
gettimeofday:
这个API也能返回一个描述当前时间的值
使用gettimeofday获得的值是Unix time。
Unix time是以UTC 1970年1月1号 00:00:00为基准时间,当前时间距离基准点偏移的秒数。上述API返回的值是1481266031,表示当前时间距离UTC 1970年1月1号 00:00:00一共过了1481266031秒。
这个API返回的值会受系统时间影响,用户如果修改时间,值也会随着变化。

sysctl
返回的值是上次设备重启的时间Unix time。(eg上次开启时间是10.28号,则返回的时间就是1970年到开启那天的unix时间)
这个API返回的值也会受系统时间影响,用户如果修改时间,值也会随着变化。

  • (long)bootTime
    {

define MIB_SIZE 2

int mib[MIB_SIZE];
size_t size;
struct timeval  boottime;

mib[0] = CTL_KERN;
mib[1] = KERN_BOOTTIME;
size = sizeof(boottime);
if (sysctl(mib, MIB_SIZE, &boottime, &size, NULL, 0) != -1)
{
    return boottime.tv_sec;
}
return 0;

}

mach_absolute_time()
CPU的时钟周期数(ticks)。这个tick的数值可以用来描述时间,而mach_absolute_time()返回的就是CPU已经运行的tick的数量。将这个tick数经过一定的转换就可以变成秒数,或者纳秒数,这样就和时间直接关联了。
这个tick数,在每次手机重启之后,会重新开始计数,而且iPhone锁屏进入休眠之后tick也会暂停计数。
mach_absolute_time()不会受系统时间影响,只受设备重启和休眠行为影响

获取系统当前运行了多长时间:
gettimeofday和sysctl都会受系统时间影响,但他们二者做一个减法所得的值,就和系统时间无关了。这样就可以避免用户修改时间了

  • (NSTimeInterval)uptime
    {
    struct timeval boottime;
    int mib[2] = {CTL_KERN, KERN_BOOTTIME};
    size_t size = sizeof(boottime);
    struct timeval now;
    struct timezone tz;
    gettimeofday(&now, &tz);
    double uptime = -1;
    if (sysctl(mib, 2, &boottime, &size, NULL, 0) != -1 && boottime.tv_sec != 0)
    {
    uptime = now.tv_sec - boottime.tv_sec;
    uptime += (double)(now.tv_usec - boottime.tv_usec) / 1000000.0;
    }
    return uptime;
    }

参考博客:https://juejin.im/post/5bf121b6f265da613a539d9c
https://mp.weixin.qq.com/s/cSZUNMuqk6DL3-nctyxzcw

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,303评论 19 139
  • 本文包括:1、Listener简介2、Servlet监听器3、监听三个域对象创建和销毁的事件监听器4、监听三个域对...
    廖少少阅读 6,495评论 6 28
  • 我们在开发应用的过程中,往往在很多地方需要倒计时,比如说轮播图,验证码,活动倒计时等等。而在实现这些功能的时候,我...
    伯陽阅读 3,105评论 3 22
  • 目录 上一章:惊魂一夜 从古至今,无论城镇乡村,建房都是人们生活中的一件大事,建房中的礼仪其实是一种求吉、祈福仪式...
    人鱼海棠阅读 532评论 11 9
  • 不知道是不是最近太累的原因,总之自己没觉得有什么,身体却开始抗议了!大姨妈已经推迟了快二十天,每天失眠,身体疲倦,...
    花儿与月之歌阅读 369评论 0 0

友情链接更多精彩内容