经常出现的8小时
在代码中写到关于时间的问题时经常遇到一个惹人烦的8小时问题,这个就是因为我们是在东八区 ,而有时候后端返给我们的时间是零时区的时间,东八区比零时区要早8小时,比如现在东八区是中午十二点,零时区就是凌晨四点,那么我们就需要把零时区的时间转化为东八区的时间。
首先看一个例子,获取当前时间字符串:
//通过[NSDate date]返回的一定是零时区的时间
NSDate *date = [NSDate date];
NSLog(@"date时间 = %@", date);
//如果没有规定formatter的时区,那么formatter默认的就是当前时区,比如现在在北京就是东八区,在东京就是东九区
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
//最结尾的Z表示的是时区,零时区表示+0000,东八区表示+0800
[formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss Z"];
使用formatter转换后的date字符串变成了当前时区的时间
NSString *dateStr = [formatter stringFromDate:date];
NSLog(@"字符串时间 = %@", dateStr);
输出:
2018-05-31 14:43:07.011223+0800 TimeDemo[4674:167771] date时间 = 2018-05-31 06:43:07 +0000
2018-05-31 14:43:07.011777+0800 TimeDemo[4674:167771] 字符串时间 = 2018-05-31 14:43:07 +080
很重要的一点就是,[NSDate date]返回的是零时区的时间!返回的是零时区的时间!返回的是零时区的时间!
打印结果前面的时间是北京时间,为2018-05-31 14:43:07.011223,而通过[NSDate date]得到的时间是2018-05-31 06:43:07 +0000,后面的+0000就表示了这个时间是零时区的时间,这个时间对应的东八区的时间就是2018-05-31 14:43:07.011223 +0800了。这两个时间是同一个时间,只不过时区不一样。
formatter在不特意设置时区的情况下,默认时区是当前地区的时区,比如我们如果在中国,当前时区就是东八区,我们如果把手机拿到东京,当前时区就是东九区。经过formatter转化为时间字符串后时间就变成了当前时区也就是东八区的时间2018-05-31 14:43:07 +080,也就没有了8小时的误差。
转字符串时间的时区设置
上面例子中NSDate时间转字符串时间时,NSDateFormatter并没有设置时区,而是使用当前地区的时区,这和设置系统系统时区formatter.timeZone = [NSTimeZone systemTimeZone]的效果是一样的。我们也可以规定一定使用某一个时区:
NSDate *date = [NSDate date];
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
[formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss"];
formatter.timeZone = [NSTimeZone timeZoneWithName:@"Asia/Shanghai"];//东八区时间
NSString *dateStr = [formatter stringFromDate:date];
NSLog(@"字符串时间 = %@", dateStr);
这样不管我们的手机是在哪里,打印出来的时间都是东八区的时间。
还有如下的时区指定:
formatter.timeZone = [NSTimeZone timeZoneWithName:@"Asia/Tokyo"];//东九区时间
formatter.timeZone = [NSTimeZone timeZoneWithName:@"GMT"];//零区时间
formatter.timeZone = [NSTimeZone timeZoneWithName:@"UTC"];//零区时间,和GMT一样
字符串时间转NSDate
字符串时间转NSDate也会有时间问题,看下面的例子:
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
[formatter setDateFormat:@"yyyy-MM-dd HH:mm:ss Z"];
NSDate *newDate = [formatter dateFromString:@"2016-12-07 14:06:24 +0800"];
NSLog(@"newDate = %@", newDate);
打印结果:
2018-05-31 15:22:11.878915+0800 TimeDemo[5236:189492] newDate = 2016-12-07 06:06:24 +0000
我们看到这样转化后就出现了问题转后之后得到的时间的时区又变成了零时区。问题就出在,凡是返回结果为NSDate类型的,得到的时间都是零时区,都是零时区!都是零时区!
如果上面给出的字符串时间是@"2016-12-07 14:06:24 +0000",也就是这个时间字符串的时区是零时区,那么得到的结果就不会有变化。
NSDateFormatter的格式为@"yyyy-MM-dd HH:mm:ss Z",这里面的Z表示的就是时区,我们在传递时间字符串的时候也要按照这个规则在时间后面加上+0800或者+0000这样表示时区的。
当不指定字符串的时区时,即NSDateFormatter的格式为@"yyyy-MM-dd HH:mm:ss",这样传递进来的字符串的时区默认为当前时区,转化为NSDate是零时区时间。
当前时间转时间戳
时间戳是指1970年1月1日0时0分0秒到当前时间的秒数。注意这里的当前时间是指零时区的NSDate时间。
NSDate *date = [NSDate date];
NSTimeInterval timeIn = [date timeIntervalSince1970];
NSLog(@"时间戳 = %.0f", timeIn);
打印结果:
2018-05-31 15:49:48.165702+0800 TimeDemo[5635:205558] 时间戳 = 1527752988
NSDate转当前时区的NSDate时间
NSDate *date = [NSDate date];
NSTimeZone *zone = [NSTimeZone systemTimeZone];
//获得当前时区和零时区的秒数偏移量
NSInteger interval = [zone secondsFromGMTForDate:date];
NSDate *localDate = [date dateByAddingTimeInterval:interval];
NSLog(@"localDate = %@",localDate);
时间戳和时间字符串的相互转换
时间戳转时间字符串
- (NSString *)convertToTimeStringWithTimeInterval:(NSTimeInterval)timeInterval{
NSDate *date = [NSDate dateWithTimeIntervalSince1970:timeInterval];
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setDateFormat:@"yyyy-MM-dd HH:mm:ss Z"];
NSString *dateStsring = [dateFormatter stringFromDate:date];
return dateStsring;
}
时间字符串转时间戳
- (NSTimeInterval)convertToTimeIntervalWithTimeString:(NSString *)timeString{
NSDateFormatter *dateFormatter = [[NSDateFormatter alloc] init];
[dateFormatter setDateFormat:@"yyyy-MM-dd HH:mm:ss Z"];
NSDate *date = [dateFormatter dateFromString:timeString];
NSTimeInterval timeInterval = [date timeIntervalSince1970];
return timeInterval;
}