在 Swift 2 中使用 NSDateComponents 计算下个月开始的奇怪日期结果
Strange date result calculating start of next month using NSDateComponents in Swift 2
我有一个函数:
internal func startOfNextMonth() -> NSDate? {
guard
let cal: NSCalendar = NSCalendar.currentCalendar(),
let comp: NSDateComponents = NSDateComponents() else { return nil }
comp.month = 1
comp.day = 0
comp.to12pm()
let date = cal.dateByAddingComponents(comp, toDate: self.startOfMonth()!, options: [])!
return date
}
这应该会计算下个月的第一天,但它会返回 2016-03-02 08:00:00 UTC
。
这是否与闰年有关(2 月 29 日搞砸了?)
这是我的 startOfMonth
函数供参考,to12pm()
扩展:
internal func startOfMonth() -> NSDate? {
guard
let cal: NSCalendar = NSCalendar.autoupdatingCurrentCalendar(),
let comp: NSDateComponents = cal.components([.Year, .Month], fromDate: self.at12pm()!) else { return nil }
comp.to12pm()
return cal.dateFromComponents(comp)!
}
internal extension NSDateComponents {
func to12pm() {
self.hour = 12
self.minute = 0
self.second = 0
}
}
internal func at12pm() -> NSDate? {
let cal = NSCalendar.currentCalendar()
return cal.dateBySettingHour(12, minute: 0, second: 0, ofDate: self, options: [])
}
如果你只想计算下个月的第一天,你所要做的就是:
let calendar = NSCalendar.autoupdatingCurrentCalendar()
let todayComponents = calendar.components([.Year, .Month], fromDate: NSDate())
todayComponents.month += 1
let firstDayOfMonth = calendar.dateFromComponents(todayComponents)
测试:
print(firstDayOfMonth)
在控制台中打印:
Optional(2016-03-01 08:00:00 +0000)
另外,如果是12月,日期会溢出到下一年的1月。
startOfMonth()
returns 下个月第一天中午 12 点 (12:00) 然后你在 startOfNextMonth()
中再添加 12 小时,结果提前一天。
NSCalendar
有一个聪明的方法来计算下个月的开始,而不管夏令时的变化或其他不规则性
let calendar = NSCalendar.currentCalendar()
let components = NSDateComponents()
components.day = 1
let startOfNextMonth = calendar.nextDateAfterDate(NSDate(), matchingComponents: components, options: .MatchNextTime)
我有一个函数:
internal func startOfNextMonth() -> NSDate? {
guard
let cal: NSCalendar = NSCalendar.currentCalendar(),
let comp: NSDateComponents = NSDateComponents() else { return nil }
comp.month = 1
comp.day = 0
comp.to12pm()
let date = cal.dateByAddingComponents(comp, toDate: self.startOfMonth()!, options: [])!
return date
}
这应该会计算下个月的第一天,但它会返回 2016-03-02 08:00:00 UTC
。
这是否与闰年有关(2 月 29 日搞砸了?)
这是我的 startOfMonth
函数供参考,to12pm()
扩展:
internal func startOfMonth() -> NSDate? {
guard
let cal: NSCalendar = NSCalendar.autoupdatingCurrentCalendar(),
let comp: NSDateComponents = cal.components([.Year, .Month], fromDate: self.at12pm()!) else { return nil }
comp.to12pm()
return cal.dateFromComponents(comp)!
}
internal extension NSDateComponents {
func to12pm() {
self.hour = 12
self.minute = 0
self.second = 0
}
}
internal func at12pm() -> NSDate? {
let cal = NSCalendar.currentCalendar()
return cal.dateBySettingHour(12, minute: 0, second: 0, ofDate: self, options: [])
}
如果你只想计算下个月的第一天,你所要做的就是:
let calendar = NSCalendar.autoupdatingCurrentCalendar()
let todayComponents = calendar.components([.Year, .Month], fromDate: NSDate())
todayComponents.month += 1
let firstDayOfMonth = calendar.dateFromComponents(todayComponents)
测试:
print(firstDayOfMonth)
在控制台中打印:
Optional(2016-03-01 08:00:00 +0000)
另外,如果是12月,日期会溢出到下一年的1月。
startOfMonth()
returns 下个月第一天中午 12 点 (12:00) 然后你在 startOfNextMonth()
中再添加 12 小时,结果提前一天。
NSCalendar
有一个聪明的方法来计算下个月的开始,而不管夏令时的变化或其他不规则性
let calendar = NSCalendar.currentCalendar()
let components = NSDateComponents()
components.day = 1
let startOfNextMonth = calendar.nextDateAfterDate(NSDate(), matchingComponents: components, options: .MatchNextTime)