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;
}


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容

  • iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等。解决办法网上比比皆是,但大多...
    小李龍彪阅读 6,291评论 1 6
  • 在iOS开发中,经常会遇到各种各样的时间问题,8小时时差,时间戳,求时间间隔,农历等等。解决办法网上比比皆是,但大...
    真巧了_嘿阅读 2,754评论 0 7
  • 在Python中,迭代是通过for ... in来完成的 默认情况下,dict迭代的是key。如果要迭代value...
    牛崽儿酷阅读 98评论 0 0
  • 晚上去看了学校里的红楼梦音乐会。说来惭愧,《红楼梦》我还没读完,高中时候由于是推荐阅读书目,我便买了回来,看了三四...
    咯子嘣阅读 263评论 0 0