美文网首页
Object-C时区和时间 2023-03-25 周六

Object-C时区和时间 2023-03-25 周六

作者: 勇往直前888 | 来源:发表于2023-03-24 18:02 被阅读0次

    简介

    生产线报了一个问题,就是照片的水印时间和服务器的操作时间对不上:
    照片水印时间是3月21日; 而服务显示的处理时间是3月22日;

    原因

    拍照手机的时区设置成了外国。

    水印日期:

    查看了一下代码,水印日期的生成取的是手机本地时间:

        // 日期
        NSDate *now = [NSDate date]; /** 取当前的日期*/
        NSDateFormatter *formatter  = [[NSDateFormatter alloc] init];
        // 时区设置为系统默认时区
        NSTimeZone *timeZone = [NSTimeZone systemTimeZone];;
        [formatter setTimeZone:timeZone];
        [formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
        NSString *dateString = [formatter stringFromDate:now];
        dateString = [NSString stringWithFormat:@"%@ UTC+8", dateString];
    
    • 因为仓库位置在广东惠州,systemTimeZone就是东8区,所以默认情况下,就是UTC+8

    • 现在手机时区设置到了外国,那么systemTimeZone就不是东8区了,那么所得时间字符串就和UTC+8存在很大的差距

    如果解决时区问题?

    • 检查仓库的拍照手机,确保时区都是默认的东8区(也就是北京时间),这当然没问题

    • 那有没有办法,就算时区设置错了,水印时间仍旧是东8区呢?
      当然是可以的,按照字面意思修改就是了:

        // 日期
        NSDate *now = [NSDate date]; /** 取当前的日期*/
        NSDateFormatter *formatter  = [[NSDateFormatter alloc] init];
        // 时区设置为UTC+8
        NSTimeZone *timeZone = [NSTimeZone timeZoneForSecondsFromGMT:(8 * 60 * 60)];
        [formatter setTimeZone:timeZone];
        [formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
        NSString *dateString = [formatter stringFromDate:now];
        dateString = [NSString stringWithFormat:@"%@ UTC+8", dateString];
    

    通过代码调试跟踪可以知道NSDate *now = [NSDate date];这句话得到的时间是UTC时间
    由于这里要的就是UTC+8,所以按照字面意思,时区设置为[NSTimeZone timeZoneForSecondsFromGMT:(8 * 60 * 60)]

    经过这样修改之后,不论手机本身的时区设置是什么,水印图片上的时区都是东8区(北京时间),问题得到时间。

    延伸问题:改了时间怎么办?

    • 修改时区,可以通过固定时区的方式解决,但是修改了时间,那么时间就是错的。而且修改时间很随意,可以差几天,几小时,也可以差几分钟,几秒钟,没办法规范。

    • 后来想到的办法获取服务器时间,和本地时间做对比,就可以知道本地时间是否正确。

    那么如何获得服务器时间呢?

    • 百度上好几篇文章介绍的方法是访问百度,然后从header中取日期字段,比如下面这篇:
      iOS获取网络时间,网络获取时间,也就是现实中的时间

    • 看了文章,发现代码很老,并且好多函数已经被废弃。也不知道为什么都是这种低质量的文章,非常失望。

    • 虽然文章质量很差,也引入工程试了一下,结果是:没效果。

    有效的文章

    服务器响应时间
    • 看时间字符串的格式,应该是UTC时间"Sat, 25 Mar 2023 09:44:10 GMT"

    计算时间差

    • 将类似"Sat, 25 Mar 2023 09:44:10 GMT"的字符串转化为NSDate,然后与手机本地时间比较一下,就可以知道本地时间是否准了。(默认服务器时间是准的,手机本地时间是不准的)

    • 如何衡量准不准?当时是做减法,用服务器时间-本地时间,看差值就可以了。既然AFNetworking返回的是UTC时间,本地时间直接用UTC时间就好了(省事,不要设置NSTimeZone就好了,默认就是UTC时间)

    • 时间差代码如下:

    /// 计算服务器时间与本地时间的差值
    /// https://github.com/eye1234456/CorrectLocalTime
    + (NSTimeInterval)localDifferenceWithResponse:(NSURLResponse * __nullable)response {
        // 取服务器响应的Date字段
        if (![response isKindOfClass:NSHTTPURLResponse.class]) {
            return 0;
        }
        NSHTTPURLResponse *httpResponse = (NSHTTPURLResponse *)response;
        NSString *dateServer = [httpResponse.allHeaderFields objectForKey:@"Date"];
        if (dateServer == nil) {
            return 0;
        }
        if (![dateServer isKindOfClass:NSString.class]) {
            return 0;
        }
        
        // 将字符串转化为NSDate对象, UTC时间
        // dateServer类似Sun, 19 May 2002 15:21:36 GMT
        NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
        [dateFormatter setTimeZone:[NSTimeZone timeZoneForSecondsFromGMT:0]];
        [dateFormatter setDateFormat:@"EEE, d MMM yyyy HH:mm:ss zzz"];
        NSDate *inputDate = [dateFormatter dateFromString:dateServer];
        if (inputDate == nil) {
            return 0;
        }
        
        // 本地时间
        NSDate *localDate = [NSDate date];
        
        // 服务器时间与本地时间之间的误差
        NSTimeInterval distance = [inputDate timeIntervalSinceDate:localDate];
        
        return distance;
    }
    

    如何使用?

    • 首先确定准不准的标准,比如5分钟,也就是(5 * 60)秒

    • 其次,时间差有正有负,只要绝对值就可以了,用个fabs()就可以了

    • 另外,网络接口不处理界面,只要在检测到误差过大(这里是超过5分钟),通过消息的形式传出去就可以了

                // 计算时间差
                NSTimeInterval difference = [PDANetwork localDifferenceWithResponse:response];
                NSTimeInterval tolerance = 5 * 60; // 5 分钟
                if (fabs(difference) > tolerance) {
                    NSLog(@"时间误差过大");
                    /// 本地日期时间被修改消息
                    [[NSNotificationCenter defaultCenter] postNotificationName:PDALocalDateTimeChangeNotification object:nil userInfo:nil];
                }
    

    如何处理?

    • 最简单的方案就是在首页接收消息,检测到误差过大,弹窗提示,并且退出程序。
    - (void)LocalDateTimeChange:(NSNotification *)notifacation {
        /// 弹窗提示,并退出程序
        UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"本地日期时间被修改" message:@"本地时间和服务器时间误差超过5分钟,请退出调整时间设置" preferredStyle:UIAlertControllerStyleAlert];
        [alert addAction:[UIAlertAction actionWithTitle:@"知道了" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
            exit(0);
        }]];
        
        [self presentViewController:alert animated:YES completion:nil];
    }
    
    • 当然也可以把时间差作为公共参数存在本地缓存,在用到时间的时候进行补偿计算;
      在需要的时候可以这样做。
      当然,在仓库这种场合,这种乱设时间的做法是不允许的,还是直接退出程序不让用来得更好。

    相关文章

      网友评论

          本文标题:Object-C时区和时间 2023-03-25 周六

          本文链接:https://www.haomeiwen.com/subject/wbtgrdtx.html