简介
生产线报了一个问题,就是照片的水印时间和服务器的操作时间对不上:
照片水印时间是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获取网络时间,网络获取时间,也就是现实中的时间 -
看了文章,发现代码很老,并且好多函数已经被废弃。也不知道为什么都是这种低质量的文章,非常失望。
-
虽然文章质量很差,也引入工程试了一下,结果是:没效果。
有效的文章
-
继续百度,终于找到了有效的文章:
iOS通过服务器时间校验防止时间篡改出问题 -
并且在gitHub上有demo,可以看源代码,非常好:
CorrectLocalTime -
看了一下文章,原来是 AFNetworking会返回服务器的时间,放在response的Header种,字段是Date
-
项目中网络用的就是AFNetworking,所以直接断点跟踪一下就可以验证
- 看时间字符串的格式,应该是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];
}
- 当然也可以把时间差作为公共参数存在本地缓存,在用到时间的时候进行补偿计算;
在需要的时候可以这样做。
当然,在仓库这种场合,这种乱设时间的做法是不允许的,还是直接退出程序不让用来得更好。
网友评论