美文网首页
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];
}
  • 当然也可以把时间差作为公共参数存在本地缓存,在用到时间的时候进行补偿计算;
    在需要的时候可以这样做。
    当然,在仓库这种场合,这种乱设时间的做法是不允许的,还是直接退出程序不让用来得更好。

相关文章

  • 查看/修改LINUX时区和时间

    查看/修改LINUX时区和时间,更新系统时间 一、时区1. 查看当前时区date -R2. 修改设置时区方法(1)...

  • 时间序列

    时区表示 时区转换 时间跨度转换 时期和时间戳之间的转换

  • 时间戳、时间和时区

    时间戳 时间戳是容易理解的。 时间戳是指格林威治时间1970年01月01日00时00分00秒(北京时间1970年0...

  • Android实践 -- 设置系统日期时间和时区

    设置系统日期时间和时区 设置系统的日期时间和时区,需要 系统权限和系统签名,android:sharedUserI...

  • 时间格式和时区

    /* G: 公元时代,例如AD公元 yy: 年的后2位 yyyy: 完整年 MM: 月,显示为...

  • 时间戳和时区

    Unix时间戳(Unix timestamp),Unix时间(Unix time)、druid POSIX时间(P...

  • 时区那些事儿

    为什么会有时区?一共有多少个时区?时区与时间的关系?哪个时区的时间更快一些?东十二区和西十二区是不是同一个时区?东...

  • node 和 mysql 里时区的理解

    mysql 里的 timestamp 这个会根据系统时间和时区, 与数据的时间和时区进行计算. 比如系统时间是 2...

  • centos中修改docker容器的时区

    查看dock二容器时间及时区 如果发现docker容器里面的时区和时间不符合当地的实际情况,那么: 直接复制时区文...

  • Android中的时区TimeZone

    设置系统日期时间和时区 设置系统的日期时间和时区,需要 系统权限和系统签名, 需要在manifest文件中添加相应...

网友评论

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

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