NSCalendar dateByAddingComponents如何处理夏令时出错

时间:2013-11-04 18:36:25

标签: dst nscalendar

我很难理解dateByAddingComponents如何处理夏令时的异常。我已阅读Apple日期和时间编程指南,并期望dateByAddingComponents考虑DST更改。但是,在DST更改之日,它不适用于我。

以下是代码:

NSCalendar *gregorian = [[NSCalendar alloc] initWithCalendarIdentifier:NSGregorianCalendar];
[gregorian setTimeZone:[NSTimeZone localTimeZone]];
NSDateComponents *midnight = [gregorian components:(NSYearCalendarUnit | NSMonthCalendarUnit |  NSDayCalendarUnit | NSHourCalendarUnit | NSMinuteCalendarUnit | NSSecondCalendarUnit) fromDate:self.currentDate];
midnight.hour = 0;
midnight.minute = 0;
midnight.second = 0;


NSDate *startDate = [gregorian dateFromComponents:midnight];
NSDateComponents *offSetComponents = [[NSDateComponents alloc] init];
[offSetComponents setDay:1];
NSDate *endDate = [gregorian dateByAddingComponents:offSetComponents toDate:startDate options:0];

//Calculate start time from config (hours/min from seconds)
int startTimeInMinutes = self.club.clubConfiguration.startTime.integerValue;
int startTimeHours = startTimeInMinutes / 60;
int startTimeMins = startTimeInMinutes % 60;
NSLog(@"---- startTimeHours %i", startTimeHours);
NSLog(@"---- startTImeMins %i", startTimeMins);

NSDateComponents *offSetComponents2 = [[NSDateComponents alloc] init];
[offSetComponents2 setHour:startTimeHours];
[offSetComponents2 setMinute:startTimeMins];
NSDate *firstTeeTime = [gregorian dateByAddingComponents:offSetComponents2 toDate:startDate options:0];

说明: 我从用于计算firstTeeTime的服务器获取startTimeInMinutes。例如,我期望在startDate(在我的用例中是12am)增加6个小时,并获得6am(localTimeZone)。

使用dateByAddingComponents在DST更改之前和之后都可以工作,但是在DST更改当天11月3日的那天,我已经到了凌晨5点。

理论:由于11月3日星期日实际上有2个凌晨2点,我可能要考虑到这一点?如果是这种情况,我必须编写一些逻辑来说明DST更改的实际日期,并在适当时使用daylightSavingTimeOffsetForDate添加偏移量。

我缺少什么?

编辑:好的,我决定通过确定今天是否是DST更改并添加/删除一小时偏移来解决此问题。感觉有点像我在这里遗漏了一些关于NSDate的东西,但是,它有效。希望这能帮助其他人整个早上都在挠头。

解决代码:

    ////// Work around for DST
NSTimeZone *currentZone = [gregorian timeZone];
NSDate *dstTransitionDate = [currentZone nextDaylightSavingTimeTransitionAfterDate:startDate];
NSTimeInterval dstOffset = [currentZone daylightSavingTimeOffsetForDate:endDate];
NSDateComponents *startDateComponents = [gregorian components:(NSYearCalendarUnit | NSMonthCalendarUnit | NSDayCalendarUnit) fromDate:startDate];
NSDateComponents *dstTransitionDateComponents = [gregorian components:(NSYearCalendarUnit | NSMonthCalendarUnit | NSDayCalendarUnit) fromDate:dstTransitionDate];
int offset = 0;

if ( [startDateComponents year] == [dstTransitionDateComponents year] &&
        [startDateComponents month] == [dstTransitionDateComponents month] &&
        [startDateComponents day] == [dstTransitionDateComponents day])
{
    if (dstOffset > 0){
       offset = -1;
    } else {
        offset = 1;
    }
}
//////

1 个答案:

答案 0 :(得分:1)

我完全同情你,因为我陷入了同样的陷阱。首先,我对结果和文档略有不满,然后通过尝试推出自定义"添加日期组件"来尝试愚蠢的差事。逻辑。

然后,遇到了你的答案,它的效果非常好,我的测试终于通过了:

// assert!
XCTAssertEqual(dateFormatter.stringFromDate(dstSwitch.date), "11/1/15, 12:00 AM")

// Offseting the date should just work
do {
    let dstOffset = dstSwitch + 3.hours
    XCTAssertEqual(dateFormatter.stringFromDate(dstOffset.date), "11/1/15, 3:00 AM")
}

......但坚持下去! Words of wisdom from @RobNapier让我走上了正确的道路......我们正在增加6个小时,这意味着活动中的人只会花费6个小时的生命。如果开始日期是午夜,他们将在凌晨2点将手表调整回到凌晨1点,因此根据日历将丢失的时间用于遗忘。

所以...如果活动真正在午夜开始,并在早上6点结束,请注意Rob的话,不要使用持续时间。使用确切的日期组件。因为持续时间实际上是7个小时。

为了好玩,当我的测试困扰我时,我开始意识到这一点。为什么我只应在同一天添加偏移?为什么改变日子"只是工作"? ...... Liiiight bulb

而且,顺便说一句,我的测试是针对我正在推出的库,它应该结束所有日期计算的缺点并结束混乱。使用非常简洁的swift语法。 Will be available as Datez, part of Kitz