计算数组中特定月份的日期数

时间:2013-07-24 15:48:12

标签: objective-c cocoa nsdate date-arithmetic

我有一组NSDate个对象,我想知道每个月有多少条目,从现在到五个月前。

这可以使用NSPredicate吗?

2 个答案:

答案 0 :(得分:2)

与Matthias有点不同的方法,虽然我也没有看到用NSPredicate做任何好方法,因为这里需要日期算术。

这会从最初的每个日期创建日期组件,并使用NSCountedSet计算日期组件。 (您可以在计数集上使用谓词。)然后创建另一个NSDateComponents对象,描述您感兴趣的月份和年份,并使用countForObject:

查询集合
NSArray * dates = @[/* Your dates here */];

NSCalendar * cal = [NSCalendar currentCalendar];

/* Convert the dates into date components objects for month and year. */
NSMutableArray * months = [NSMutableArray array];
for( NSDate * date in dates ){

    NSDateComponents * month = [cal components:(NSMonthCalendarUnit|NSYearCalendarUnit)
                                      fromDate:date];
    [months addObject:month];
}

/* Count the unique components. */
NSCountedSet * monthsCounts = [NSCountedSet setWithArray:months];

/* Use a predicate here on the set? */

/* A components object for month arithmetic. */
NSDateComponents * monthDelta = [NSDateComponents new];

/* Get month names for the current locale. */
NSDateFormatter * formatter = [NSDateFormatter new];
NSArray * monthNames = [formatter monthSymbols];

for( NSInteger i = 0; i > -NUM_MONTHS; i-- ){

    /* Couting backwards from the current month, get a date components
     * object for each month and year.
     */
    [monthDelta setMonth:i];
    NSDateComponents * currMonth;
    currMonth = [cal components:(NSMonthCalendarUnit|NSYearCalendarUnit)
                       fromDate:[cal dateByAddingComponents:monthDelta
                                                     toDate:[NSDate date]
                                                    options:0]];
    /* Get the count from the counted set and display. */
    NSUInteger count = [monthsCounts countForObject:currMonth];
    NSLog(@"%lu dates in %@", count,
          [monthNames objectAtIndex:[currMonth month] - 1]);
}

答案 1 :(得分:1)

您必须测试数组中的日期6次。

第一:匹配7月1日或之后和8月1日之前的所有日期 第二:匹配6月1日或之后和7月1日之前的所有日期 第三:匹配5月1日或之后和6月1日之前的所有日期 等等

在循环中创建两个日期。第一个日期是午夜的月初。第二个日期恰好是1个月(!)之后。然后,过滤掉第一个日期之前或之后的数组中的所有日期。在循环的下一次迭代中,您将startDate比前一个startDate提前一个月。

大量的日期计算,但在NSDateComponents的帮助下,这不是什么大问题。

由于我不确定如何在数组中使用带有NSDates的NSPredicate,我使用了另一种方法,但这基本上是它的工作方式:

NSDateComponents *oneMonthOffset = [[NSDateComponents alloc] init];
oneMonthOffset.month = 1;

NSDateComponents *thisMonth = [calendar components:NSYearCalendarUnit|NSMonthCalendarUnit fromDate:[NSDate date]];
thisMonth.day = 1;
NSDate *startOfCurrentMonth = [calendar dateFromComponents:thisMonth];

for (NSInteger offset = 0; offset > -5; offset--) {
    NSDateComponents *startDateOffset = [[NSDateComponents alloc] init];
    startDateOffset.month = offset;
    NSDate *startDate = [calendar dateByAddingComponents:startDateOffset toDate:startOfCurrentMonth options:0];
    NSDate *endDate = [calendar dateByAddingComponents:oneMonthOffset toDate:startDate options:0];
    NSIndexSet *matchingIndexes = [array indexesOfObjectsPassingTest:^BOOL(NSDate *date, NSUInteger idx, BOOL *stop) {
        if ([date compare:startDate] != NSOrderedAscending && [date compare:endDate] == NSOrderedAscending) {
            // date is not before startDate (i.e. is on startDate or later than startDate) and date is before endDate
            return YES;
        }
        return NO;
    }];
    NSLog(@"%d dates after %@ and before %@", [matchingIndexes count], [startDate descriptionWithLocale:[NSLocale currentLocale]], [endDate descriptionWithLocale:[NSLocale currentLocale]]);
}

如果您的日期数组非常大,您可能需要先对其进行排序,然后才扫描相关范围。此代码目前测试数组中的所有NSDates。