iOS 时间校准

今天接到一个很有意思的需求。消息需要同步,需要时间校准,防止用户恶意篡改时间或无网络情况下定位准确时间。
在网上找了下资料,很多使用了ios-ntp,个人使用后感觉并不好用。

研究了一大堆,最后确定了方案。感谢大佬提供的思路参考:https://zhuanlan.zhihu.com/p/24377367

首先是获取设备运行时间:

+ (NSTimeInterval)runTime{
    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;
}

然后是获取服务器时间的方法:

- (NSDate *)getInternetDate{
    NSArray *urlList = @[@"http://www.baidu.com",
                         @"https://www.sogou.com",
                         @"https://www.sina.com.cn"];
    NSDate *localeDate;
    for (NSString *urlString in urlList) {
        NSString *url = [urlString stringByAddingPercentEscapesUsingEncoding: NSUTF8StringEncoding];
        NSMutableURLRequest *request = [[NSMutableURLRequest alloc] init];
        [request setURL:[NSURL URLWithString: url]];
        [request setCachePolicy:NSURLRequestReloadIgnoringCacheData];
        [request setTimeoutInterval: 2];
        [request setHTTPShouldHandleCookies:FALSE];
        [request setHTTPMethod:@"GET"];
         NSHTTPURLResponse *response;
        [NSURLConnection sendSynchronousRequest:request returningResponse:&response error:nil];
        NSString *date = [[response allHeaderFields] objectForKey:@"Date"];
        date = [date substringFromIndex:5];
        date = [date substringToIndex:[date length]-4];
        NSDateFormatter *dMatter = [[NSDateFormatter alloc] init];
        dMatter.locale = [[NSLocale alloc] initWithLocaleIdentifier:@"en_US"];
        [dMatter setDateFormat:@"dd MMM yyyy HH:mm:ss"];
        NSDate *netDate = [[dMatter dateFromString:date] dateByAddingTimeInterval:0];
       NSTimeZone *zone = [NSTimeZone systemTimeZone];
       NSInteger interval = [zone secondsFromGMTForDate: netDate];
       localeDate = [netDate  dateByAddingTimeInterval: interval];
        if (localeDate != nil) break;
    }
    
    if (localeDate) {
        [[NSUserDefaults standardUserDefaults]setObject:localeDate forKey:CX_ST];
        [[NSUserDefaults standardUserDefaults]synchronize];
    }
    return localeDate;
}

说明一下原理:这里当我们进入APP时先需要获取一次服务器时间和系统运行时间,当我们需要使用准确时间的时候再去请求一下运行时间。这样用第一次请求的服务器时间+两次运行时间的差值就是现在准确的时间。可以做到无视本地时间的修改。我把计算方法整合了一下:

+ (NSTimeInterval)getRightTime{
    
    NSDate *lastServerDate = [[NSUserDefaults standardUserDefaults]objectForKey:CX_ST];
    if (!lastServerDate) {
        /*第一次拉取服务器时间*/
        NSTimeInterval nowTime =  [[self getInternetDate]timeIntervalSince1970];//获取新服务器时间并保存
        NSTimeInterval runTime =  [NSObject runTime];
        [self saveRunTime:runTime];//保存设备运行时间
        return nowTime;
    }else{
        /*第二次已有服务器时间*/
        NSString *lastRunTime = [[NSUserDefaults standardUserDefaults]objectForKey:CX_RT];//获取上次设备运行时间
        NSTimeInterval lastTime = [lastRunTime doubleValue];
        NSTimeInterval nowRunTime = [NSObject runTime];//获取当前设备运行时间并保存
        NSTimeInterval lastServerTimeIv= [lastServerDate timeIntervalSince1970];
        return lastServerTimeIv + (nowRunTime-lastTime);//返回上次服务器时间和两次设备运行时间差值的和
    }
}

这里还需要注意一个问题。。。

就是当设备重启以后系统运行时间是会重置的,这时候去做计算的话就会出问题。
这里也提供一个方法:
就是当我们离开APP时记录一下系统运行时间time1。 applicationDidEnterBackground:
当我们回来时又获取一次运行时间time2,然后比对时间。
如果系统未重启,那么time2是一定>time1的
由此可以判断我们是否需要重置一下保存在本地的运行时间和是否需要再向服务器拉取一次准确时间。

-(void)applicationDidEnterBackground:(UIApplication *)application{
    NSTimeInterval backTime = [NSObject runTime];
    NSString *backTimeString = [NSString stringWithFormat:@"%f",backTime];
    [[NSUserDefaults standardUserDefaults]setObject:backTimeString forKey:@"backTime"];
    [[NSUserDefaults standardUserDefaults]synchronize];
}

- (void)applicationWillEnterForeground:(UIApplication *)application{
    /*获取系统运行时间 如果大于之前亚后台保存的系统运行时间 则不重置,反之,则认为设备重启,重置 */
    NSString *lastRunTime = [[NSUserDefaults standardUserDefaults]objectForKey:@"backTime"];//获取上次设备运行时间
    NSTimeInterval lastTime = [lastRunTime doubleValue];
    NSTimeInterval enterTime = [NSObject runTime];
    if (enterTime < lastTime) {
        [NSObject resetTimeCount];
    }
}

如果是全程无网乱改时间的情况,例如IM,目前的想法是可以对照之前的已经发送的消息进行一个模糊时间判定,最后以实际发送成功的结果update数据。

好了,本文至此结束。

—————————————————————————————————————————————

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容