我很难理解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;
}
}
//////
答案 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