NSDate中夏令时的坑

最近线上推广项目的时候,运营反馈了几个bug,其中一个就是字符串转NSDate对象出现nil的情况。

举个例子:
NSString *timeStr = @"1992-04-05";
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
[formatter setDateFormat:@"yyyy-MM-dd"];
NSDate * date1 = [formatter dateFromString:timeStr];
NSLog(@"timeStr to date is %@ %@", timeStr, date1);
这里的timeStr就是用户的生日,平时测试的时候,这块代码都是ok的。那么如果我们把生日换成1986-05-04,此时的date就会转化失败,为null。
经查找资料,发现这个情况是由于夏令时引起的。

夏时令(Daylight Saving Time:DST),又称“日光节约时制”和“夏令时间”,是一种为节约能源而人为规定地方时间的制度,在这一制度实行期间所采用的统一时间称为“夏令时间”。一般在天亮早的夏季人为将时间调快一小时,可以使人早起早睡,减少照明量,以充分利用光照资源,从而节约照明用电。各个采纳夏时制的国家具体规定不同。目前全世界有近110个国家每年要实行夏令时。

1986年4月,中国中央有关部门发出“在全国范围内实行夏时制的通知”,具体作法是:每年从四月中旬第一个星期日的凌晨2时整(北京时间),将时钟拨快一小时,即将表针由2时拨至3时,夏令时开始;到九月中旬第一个星期日的凌晨2时整(北京夏令时),再将时钟拨回一小时,即将表针由2时拨至1时,夏令时结束。从1986年到1991年的六个年度,除1986年因是实行夏时制的第一年,从5月4日开始到9月14日结束外,其它年份均按规定的时段施行。在夏令时开始和结束前几天,新闻媒体均刊登有关部门的通告。1992年起,夏令时暂停实行。

  • 1986年4月13日至9月14日
  • 1987年4月12日至9月13日
  • 1988年4月10日至9月11日
  • 1989年4月16日至9月17日
  • 1990年4月15日至9月16日
  • 1991年4月14日至9月15日
以上时间段都属于夏令时,但是经过多次测试,这其中有的日子是可以转化为NSDate的,目前只发现了6个日期会有问题。1991-04-14, 1986-05-04, 1987-04-12, 1989-04-16, 1990-04-151988-04-10(ps:没有做覆盖测试,可能还有更多)

解决方案:

一、使用GMT零时区
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
[formatter setTimeZone:[NSTimeZone timeZoneWithName:@"GMT"]];// 零时区
[formatter setDateFormat:@"yyyy-MM-dd"];
二、设置lenient属性
NSDateFormatter *formatter = [[NSDateFormatter alloc] init];
formatter.lenient = YES; // 这个属性没有官方的解释,个人理解为:如果当前时间不存在的话,会默认获取距离最近的整点时间
[formatter setDateFormat:@"yyyy-MM-dd"];

你可能感兴趣的:(NSDate中夏令时的坑)