4

我很难理解 dateByAddingComponents 如何处理夏令时的例外情况。我已通读 Apple Date and Time Programming Guide 和预期的 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 上增加 6 小时(在我的用例中为上午 12 点)并获得上午 6 点(localTimeZone)。

使用 dateByAddingComponents 在 DST 更改之前和之后都有效,但是,在 11 月 3 日星期日 DST 更改当天,我早上 5 点。

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

我错过了什么???

编辑: 好的,我决定通过确定今天是否是 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;
    }
}
//////
4

1 回答 1

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")
}

......但坚持下去!@RobNapier 的智慧之言让我走上了正确的道路……我们增加了 6 个小时,这意味着参加活动的人一生中只会度过六个小时。如果开始日期是午夜,他们会在凌晨 2 点将手表调回凌晨 1 点,因此根据日历将失去的时间花在遗忘中。

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


为了好玩,当我的测试困扰我时,我意识到了这一点。为什么我只能在同一天添加偏移量?为什么改变日子“有效”?...灯泡

而且,顺便说一句,我的测试是针对我正在推出的一个库,它应该结束所有日期计算的缺点并结束混乱.. 使用真正简洁的 swift 语法。将作为 Datez 提供,它是 Kitz 的一部分

于 2015-11-07T21:00:06.613 回答