Сортировать NSmutableArray, содержащий 2 типа объектов?

У меня есть NSMutableArray, который содержит 2 типа объектов.

Оба объекта содержат поле для дат (строки даты с разными форматами) с разными именами.

Какой лучший и самый быстрый способ отсортировать их все по дате? Можно ли использовать NSSortDescriptor в этой ситуации?


person aryaxt    schedule 17.08.2011    source источник


Ответы (2)


Если вы ориентируетесь на Mac OS X v10.6 или iOS4 и выше, вы можете сортировать с помощью компаратора, подобного этому

NSDateFormatter *formatter1 = ...; //create and configure date formatter for Class1
NSDateFormatter *formatter2 = ...; //create and configure date formatter for Class2


[arrayOfObjects sortUsingComparator:^NSComparisonResult(id obj1, id obj2) {
    NSDate *date1 = nil, *date2 = nil;
    //Get first objects date
    if([obj1 isKindOfClass:[Class1 class]])
    {
        date1 = [formatter1 dateFromString:[obj1 class1Date]];
    }
    else if([obj1 isKindOfClass:[Class2 class]])
    {
        date1 = [formatter2 dateFromString:[obj1 class2Date]];
    }

    //Get second objects date
    if([obj2 isKindOfClass:[Class1 class]])
    {
        date2 = [formatter1 dateFromString:[obj2 class1Date]];
    }
    else if([obj2 isKindOfClass:[Class2 class]])
    {
        date2 = [formatter2 dateFromString:[obj2 class2Date]];
    }

    NSAssert(date1 != nil, @"Could not parse date from %@", obj1);
    NSAssert(date2 != nil, @"Could not parse date from %@", obj2);

    return [date1 compare:date2];
}];

Помимо проверки класса, вы также можете проверить, отвечает ли он на селектор.

person Joe    schedule 17.08.2011
comment
Приведение объектов из id совершенно не нужно. - person Chuck; 18.08.2011
comment
Да, в NSArray есть несколько методов sortUsing..., и любой из них должен работать. - person Hot Licks; 18.08.2011
comment
Кстати, нет необходимости запускать isKindOfClass: или respondsToSelector:, если вы используете категории. - person nielsbot; 29.08.2012

Вы можете добавить категорию, чтобы унифицировать имя поля даты. Скажем, экземпляры класса A имеют поле "dateString", а экземпляры класса B имеют поле "creationDateString"... Вы можете сделать это:

@implementation A (SortByDate)

-(NSDate*)sortDate
{
    static NSDateFormatter * formatter = nil ;
    static dispatch_once_t once = 0 ;
    dispatch_once( & once, ^{ formatter = ... } ) ;

    return [ formatter dateFromString:self.dateString ] ;
}
@end

@implementation B (SortByDate)
-(NSDate*)sortDate
{
    static NSDateFormatter * formatter = nil ;
    static dispatch_once_t once = 0 ;
    dispatch_once( & once, ^{ formatter = ... } ) ;

    return [ formatter dateFromString:self.creationDateString ] ;
}
@end

Теперь вы можете отсортировать все объекты в вашем массиве по их свойству sortDate:

NSArray * sortedByDateArray = [ unsortedArray sortedArrayUsingComparator:^( id a, id b ){
    return [ [ a sortDate ] compare:[ b sortDate ] ] ;
}];

EDIT Или с дескриптором сортировки:

NSArray * sortedArray = [ unsortedArray sortedArrayUsingDescriptors:[ NSArray arrayWithObject:[ NSSortDescriptor sortDescriptorWithKey:@"sortDate" ascending:YES ] ] ] ;
person nielsbot    schedule 28.08.2012
comment
вы даже можете создать хранилище для каждого экземпляра для вашего нового свойства sortDate, чтобы его нужно было пересчитать только с изменениями базовой строки. - person nielsbot; 29.08.2012