NSDate 8小时问题-没你想的那么简单!

说在前面

公司项目出了问题之后,上网差了很多资料,最后就有一个还是比较靠谱,剩下的都是说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)

10:36分,北京时间
- (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.4 dateFormat转换,公式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这种方式!)


彩蛋放送

.h文件

/**
 *  将0时区的时间转成0时区的时间戳(10位数)
 */
+ (NSString *)transformToTimestampWithDate:(NSDate *)date;

/**
 *  将0时区的时间戳(10位数)转成0时区的时间
 */
+ (NSDate *)transformToDateWithTimestamp:(NSString *)timestamp;

/**
 *  将0时区的时间戳(10位数)转成8时区的时间文本格式(“2015-12-13 13:34:45”)
 */
+ (NSString *)transformToStringWithTimestamp:(NSString *)timestamp;

/**
 *  将0时区的时间戳(10位数)转成8时区的时间文本格式(“2012-12-12 12:12”),带有只有时分的
 */
+ (NSString *)transformToHourMiniteFormatWithTimestamp:(NSString *)timestamp;

/**
 *  将8时区的时间文本格式(“2015-12-13 13:34:45”)转成 0时区的时间戳(10位数)
 */
+ (NSString *)transformToTimestampWithString:(NSString *)string;

/**
 *  将8时区的时间文本格式(“2015-12-13 13:34:45”)转成 0时区的NSDate
 */
+ (NSDate *)transformToDateWithString:(NSString *)string;

/**
 *  将0时区的NSDate转成 8时区的时间文本格式(“2015-12-13 13:34:45”)
 */
+ (NSString *)transformToStringWithDate:(NSDate *)date;

.m文件

/**
 *  将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;
}


推荐阅读更多精彩内容

  • iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等。解决办法网上比比皆是,但大多...
    小李龍彪阅读 2,513评论 1 3
  • 在iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等。解决办法网上比比皆是,但大...
    真巧了_嘿阅读 1,188评论 5 7
  • 框架: 就是系统(苹果)或者第三方(其他的一些高手)事先写好了一些很牛X功能的类.把这些类交给我们使用.这些类的集...
    指尖书法阅读 1,040评论 0 8
  • NSDate -- 表示一个绝对的时间点 NSTimeZone -- 时区信息 NSLocale -- 本地化信息...
    像小强一样活着阅读 2,960评论 6 16
  • 在Python中,迭代是通过for ... in来完成的 默认情况下,dict迭代的是key。如果要迭代value...
    牛崽儿酷阅读 35评论 0 0