美文网首页iOS开发杂货铺iOS技术IOS开发路上的故事
NSDate 8小时问题-没你想的那么简单!

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

作者: mkb2 | 来源:发表于2016-08-24 23:59 被阅读5872次
说在前面

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


相关文章

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

    说在前面 公司项目出了问题之后,上网差了很多资料,最后就有一个还是比较靠谱,剩下的都是说8小时,太肤浅,今天将这些...

  • 客服没你想的那么简单!

    每一次外出学习都收获满满 ,这一次也不例外,周娟老师此次分享干货非常多,使我受益匪浅。感谢颇多,总结为以下几点. ...

  • 他们没你想的那么简单

    我从初中时候就爱写一些自己对于教育的看法,那时我的角色不同,我仅仅只是个受教育者,所以似乎我没有什么发言权,记录的...

  • 学习,没你想的那么简单

    学习,真的没你想的那么简单。 先说说学习这个概念。学习这个概念和上学完全不同,上学是被动的过程,而学习是主动的过程...

  • 整理,没你想的那么简单

    这两天一直在整理家,丢掉了很多东西,有一些新的感悟想要和大家分享。 从最开始的衣橱整理,到厨房、卫生间、最后收拾了...

  • 努力,没你想的那么简单

    如果恰好失眠,如果半夜恰好手机振动,而我想也不用想,就知道肯定是那几个“拼命三娘”刚发完文。自从加入了写作群,让我...

  • 花钱,没你想的那么简单

    你会花钱吗?这个问题可能会让很多人笑出来:钱谁不会花啊?尤其对于女性来说,买买买就解决了。可是买东西就是会花钱吗?...

  • 开车?没你想的那么简单

    近日,在网上流传出一些令人匪夷所思的视频。一些骑着电车的市民在过红绿灯路口时全部下车并推着电车过红绿灯,然后视频讲...

  • 孩子没你想的那么简单

    在昨天之前,我对于孩子得认识还是“天使”,他们天真可爱,眼中是没有被世俗污染过明亮。可是,昨天发生的一幕幕让我始料...

  • 简言(2017.5.25)

    你想要一个熊,还是想要一个熊抱?这问题没那么简单。

网友评论

  • hwreal: NSDate *date = [NSDate date];
    NSLog(@"当前时间:%@",date);
    NSLog(@"nowdate:%@",date);
    -------------------------------------
    当前时间:2017-10-20 02:39:08 +0000
    nowdate:Fri Oct 20 10:39:08 2017

    这个真是把我搞晕了
    轻诺:不看不知道,原来这么多东西. 我试了下,第一个输出有中文的时候, 就会和第二个不一样, 貌似系统自动处理了
    mkb2:我就知道一个是零式区,确实很尴尬,
  • Nicole__Zhang:今天发现了 恶心死我了
  • 小鲁靠谱:很详细,真赞 :smile:
  • dzgdp888:。。。项目中哪有这么自己换算的,为什么不用NSTimeZone,这不应该是常识吗?
    mkb2:@dzgdp888 嗯,懂你,谢谢,我到时候再添加那就时区的语句就行,到时候就解决了那个问题:smile::smile:
    dzgdp888:@王鑫20111 只是建议啊,还是用时区自动转换时间。要不app毕竟有局限性,万一以后还要在别的国家推广呢。。。
    mkb2:我懂,实际上使用nstimezone的话,转换的也是8小时,那样更加写也行,不过我这样这,省了一句话,你说的那种,可能是date转化date我不太清楚啊,之前我们也那样使用,后来看到更方便的方法,直接获得了那个公式,更加方便
  • 阿木摄影:不错
    mkb2:谢谢
  • 代亮真的不会亮:有一个类叫做NSTimeZone。。。。。时区这种有点地理常识的都知道
  • CSCloud:写的挺好的,把各种转换封装成一个NSDate类别使用起来更方便
    mkb2:添加完毕~
    mkb2:对的,我刚刚写好工具类,马上放出来一起看看:smile::smile:
  • 西门欥雪:确实不错,之前代码显示时间和向后台传递相差8小时,测试时最好把时间戳在网页上转换一下
    mkb2:对啊,之前做的时候都懵逼了,那时候一直没好好处理,现在乱七八糟,该起了灾难啊
  • e4d93d6aae84:尼玛,曾今这8小时,恶心死我了
    mkb2:我们一直受这个困扰,现在也是,今天正在改时间,写得乱的不行了:sob::sob::sob:

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

本文链接:https://www.haomeiwen.com/subject/liposttx.html