说在前面
公司项目出了问题之后,上网差了很多资料,最后就有一个还是比较靠谱,剩下的都是说8小时,太肤浅,今天将这些问题列出,顺便给NSDate做个记录,最后po出解决公司问题的方法
项目除了什么问题?
- 1.返回的时间戳好像是差了8小时
- 2.项目中的时间分类好多,不知道那个是有用的
- 3.项目中选择了datePicker,获得了一个时间,然后显示,最后传给后台,结果时间多了8小时?
基础知识普及
1.什么是UTC? 世界标准时间,国际协调时间,简称UTC。不属于任意时区
2.啥事时间戳?就是1970.1.1 00:00:00作为标准,某个时间和他的秒数,并且NSDate必须是0时区的,UTC格式的
3.时间戳应该是10位,如果不巧碰到了13位的,代表着他计算了毫秒,只要删除剪切前十位就行了
详细的讲解NSDate,一定有你不知道的知识
一.获取当前时间 (NSDate),(NSDate -> NSString)
//1.打印当前时间
NSDate *date = [NSDate date];
NSLog(@"当前时间%@",date);
//2.打印出2011-11-12 23:10:34这种格式
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc]init];
[dateFormatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
NSString *dateStr = [dateFormatter stringFromDate:date];
NSLog(@"字符串表示1:%@",dateStr);
//3.打印出2013年12月22日 12时34分56秒这个格式
NSDateFormatter *dateFormaterA = [[NSDateFormatter alloc]init];
[dateFormaterA setDateFormat:@"yyyy年MM年dd日 HH时mm分ss秒"];
NSString *dateStrA = [dateFormaterA stringFromDate:date];
NSLog(@"%@",dateStrA);
打印结果
当前时间 :2016-08-24 14:16:47 +0000
字符串表示1:2016-08-24 22:16:47
字符串表示2:2016年08年24日 22时16分47秒
解释
- 1.
[NSDate date]
获取的数据为什么少了8小时?
是因为他获取的是零时区的时间,显示的是格林尼治的时间: 年-月-日 时:分:秒:+时区,我们在东八区,所以会有8小时的问题,系统是没有毛病的- 2.第二个打印,和第三个打印有什么异同点?
2.1 相同点:都是获取的正确的北京时间,没有8小时的问题(因为系统认为NSDate是0时区的,转换成字符串应该是当前时区的,所以没问题)
2.2 相同点:都是NSDate -> NSString,而且转成字符串的文字格式多样,主要依赖DateFormat
,但是可以随意确定DateFormat
格式,都能输出。(但是NSString -> NSDate不可以随便的转换,必须要看NSString是什么格式的,然后再去写dateFormat
,否则无效)
2.3 不同点:格式不同算吗??
二.获取北京时间,获取昨天此刻,获取明天此刻,昨天和今天的时间差 (NSDate -> NSDate)
- (void)test3{
//1.获取当前时间 零时区的时间
NSDate *date = [NSDate date];
NSLog(@"当前零时区时间 %@", date);
//2.获得本地时间 东八区 晚八个小时 以秒计时
NSDate *date1 = [NSDate dateWithTimeIntervalSinceNow:8 * 60 * 60];
NSLog(@"今天此时的时间 %@",date1);
//3.昨天此时的时间
NSDate *yesterdayDate = [NSDate dateWithTimeIntervalSinceNow:(-24 + 8) * 60 * 60];
NSLog(@"昨天此时的时间 %@",yesterdayDate);
//4.明天此刻
NSDate *tomorrowDate = [NSDate dateWithTimeInterval:24 * 60 * 60 sinceDate:date1];
NSLog(@"明天此刻的时间 %@",tomorrowDate);
//5.NSTimeInterval 时间间隔(单位是秒),double 的 typedef
//昨天此时与明天此刻的时间间隔
NSTimeInterval timeInterval = [tomorrowDate timeIntervalSinceDate:yesterdayDate];
NSLog(@"昨日和明天此刻的时间(秒) %.0f",timeInterval);
}
显示的结果
当前零时区时间 2016-08-24 14:36:11 +0000
今天此时的时间 2016-08-24 22:36:11 +0000
昨天此时的时间 2016-08-23 22:36:11 +0000
明天此刻的时间 2016-08-25 22:36:11 +0000
昨日和明天此刻的时间(秒) 172800
讲解
1.dateWithTimeIntervalSinceNow
方法是当前时间开始,加上时间戳,然后的时间,因为通过[NSDate date]
方法获取少了8小时,所以现在给加上去,获取的是NSDate类型的北京时间(注意,如果是获取NSString类型的,直接通过dateFormate就能转化好,不需要考虑8小时问题!!!)
2.[tomorrowDate timeIntervalSinceDate:yesterdayDate]
方法是获取两个NSDate类型的时间差值的
三.如何获取时间戳(NSDate -> TimeStamp)
说白了就是看看某个0时区的NSDate和1970.1.1的时间差秒数就好了
NSDate *date = [NSDate date];
NSTimeInterval timeIn = [date timeIntervalSince1970];
NSLog(@"1970年1月1日0时0分0秒至今相差 %.0f 秒", timeIn);
注意,一定是要0时区的时间,如果你获取的NSDate是东八区的时间,想去转成时间戳,一定要先减去8小时才行,否则时间戳会多出8小时!!!
打印结果是
1970年1月1日0时0分0秒至今相差 1472050117 秒
四.通过NSDateFormatter 处理NSDate和字符串的相互转换
刚刚开场就讲了NSDate -> NSString的步骤,
- 1.获取零时区的时间
- 2.通过给
dateFormat
设置任意字符串转化- 3.就可以自动变成正确的北京时间字符串!
- 4.原因,系统认为NSDate是0时区的,NSString是东八区的
下面看看将NSString -> NSDate的步骤
- 1.先获取一个时间的字符串
- 2.严格按照字符串中的时间格式,给dateFormat设置样式
- 3.获得的是一个少了8小时的NSDate
- 4.手动给NSDate添加8小时,最后得到了一个北京时间的NSDate对象
- 5.原因,系统默认字符串是东八区的,但是转化NSDate,他要搞成0时区的!
- (void)test4{
//系统会认为字符串是东八区的时间, 转乘NSDate是零时区的
NSString *dateStr = @"2016年8月24日 11时05分23秒";
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc]init];
[dateFormatter setDateFormat:@"yyyy年MM月dd日 hh时mm分ss秒"];
NSDate *date = [dateFormatter dateFromString:dateStr];
//将转换回来的对象手动加上8小时,回到北京时间
NSDate *date2 = [date dateByAddingTimeInterval:8 * 60 * 60];
NSLog(@"字符串转date: %@",date2);
}
打印结果
字符串转date: 2016-08-24 11:05:23 +0000
注意,
1.一定先看时间字符串的格式,然后给dateForamte
设置样式,不过不相符,会出问题
2.通过dateFormat获取的NSDate是0时区的,所以少了八小时,但是转时间戳是没问题的
3.要想得到正确的北京的NSDate类型对象,要加上8小时
作者,你叨逼叨的说了半天,到底要说个啥?
总结
No.1 通过[NSDate date]
获取是零时区的时间
No.2 NSDate(零时区) <-> timeStamp
No.3 系统认为NSDate应该是0时区的,NSString是东八区的
No.4dateFormat
转换,公式NSDate(0时区)<-> NSString(东八区)
No.5 项目中为什么总会出现8*60*60
这样的东西?因为他们不知道前两个公式
No.6 如何尽量少使用8*60*60
还能解决项目的问题? 凡是用到了NSDate,全部使用0时区的,因为至少转换成时间戳的时候,绝对正确(NSDate
是为了内部比较,还有就是转成时间戳,上传数据,目的不是用来显示到屏幕上),通过No.4的公式
,直接通过dateFormat
转化成北京时间字符串,截取使用即可(NSString
才是用来显示的,比较就让NSDate
去做好了)这样基本就看不到8小时了
No.7 (接No.6)有一种例外,就是通过datePicker,然后获取的应该是NSDate类型的对象,应该是北京当前的时间(东八区),如果要上传服务器的话,我们要传递的是时间戳,必须转成东八区的,所以要减去8小时才行!
最后是我们公司的问题分享,看不看都行
1.为毛时间戳转化之后,获取的NSDate少了八小时?参见No.2
2.分类中写的方法好多,到底用哪个?看完文章,对号入座吧
3.为毛线用pickerDate转成时间戳多了八小时,上传后台之后,说好的时间戳是少了8小时,怎么转变成NSDate有尼玛多了八小时?No7已经说完了,不过可以再说一遍,假设pickerDate获取的是一个NSDate(系统认为这种对象是0时区的,但是你却傻呵呵的认为这个是东八区的时间,其实真实的0时区时间应该是4:00)12:00,直接转成时间戳,(时间戳足足多了8小时),然后上传服务器,又回来了这个时间戳,(过去我们认为时间戳都是少了八小时,按照过去的*惯,将 NSDate[0时区] + 8小时 -> NSDate[东八区],)结果一定是多了8小时啊,所以争取的思路,看刚刚的总结,就是少用(NSDate -> NSdate这种方式!)
/**
* 将0时区的时间转成0时区的时间戳
*/
+ (NSString *)transformToTimestampWithDate:(NSDate *)date{
NSTimeInterval inter = [date timeIntervalSince1970];
return [NSString stringWithFormat:@"%ld", (long)inter];
}
/**
* 将0时区的时间戳转成0时区的时间
*/
+ (NSDate *)transformToDateWithTimestamp:(NSString *)timestamp{
NSTimeInterval inter = [timestamp doubleValue];
NSDate * date = [NSDate dateWithTimeIntervalSince1970:inter];
return date;
}
/**
* 将0时区的时间戳转成8时区的时间文本格式(“2015-12-13 13:34:45”)
*/
+ (NSString *)transformToStringWithTimestamp:(NSString *)timestamp{
//1.先将时间戳->NSDate
NSDate *date = [self transformToDateWithTimestamp:timestamp];
//2.将date->NSString
return [[self transformToStringWithDate:date] substringToIndex:16];
}
/**
* 将0时区的时间戳(10位数)转成8时区的时间文本格式(“2012-12-12 12:12”),带有只有时分的
*/
+ (NSString *)transformToHourMiniteFormatWithTimestamp:(NSString *)timestamp{
//1.先将时间戳->NSDate
NSDate *date = [self transformToDateWithTimestamp:timestamp];
//2.将date->NSString
return [[self transformToStringWithDate:date] substringToIndex:13];
}
/**
* 将8时区的时间文本格式(“2015-12-13 13:34:45”)转成 0时区的时间戳
*/
+ (NSString *)transformToTimestampWithString:(NSString *)string{
//1.先将NSString->NSDate
NSDate *date = [self transformToDateWithString:string];
//2.将date->timestamp
return [self transformToStringWithDate:date];
}
/**
* 将8时区的时间文本格式(“2015-12-13 13:34:45”)转成 0时区的NSDate
*/
+ (NSDate *)transformToDateWithString:(NSString *)string{
NSDateFormatter *df = [[NSDateFormatter alloc] init];
[df setLocale:[NSLocale currentLocale]];
[df setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
NSDate *date = [df dateFromString:string];
return date;
}
/**
* 将0时区的NSDate转成 8时区的时间文本格式(“2015-12-13 13:34:45”)
*/
+ (NSString *)transformToStringWithDate:(NSDate *)date{
NSDateFormatter *df = [[NSDateFormatter alloc] init];
[df setLocale:[NSLocale currentLocale]];
[df setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
NSString *string = [df stringFromDate:date];
return string;
}