在 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)