4

我有一个NSArrayController并且我想对内容进行排序,以便首先对带有英文字母的任何内容进行排序,然后对具有数字和非英文字符的任何内容进行最后排序。

例如: A, B , C ... Z, 1 , 2, 3 ... 9, 구, 결, ...

目前我只知道如何按字母顺序对项目进行排序。建议?

NSSortDescriptor *sort = [[NSSortDescriptor alloc] initWithKey:@"name" ascending:YES];
        [dataController setSortDescriptors: [NSArray arrayWithObject: sort]];
4

5 回答 5

10

您可以根据sortedArrayUsingComparator需要自定义排序算法。例如,您可以使用以下行给符号优先级:

NSArray *assorted = [@"1 2 3 9 ; : 구 , 결 A B C Z ! á" componentsSeparatedByCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
NSArray *sorted = [assorted sortedArrayUsingComparator:^NSComparisonResult(id obj1, id obj2) {
    /* NSOrderedAscending, NSOrderedSame, NSOrderedDescending */
    BOOL isPunct1 = [[NSCharacterSet punctuationCharacterSet] characterIsMember:[(NSString*)obj1 characterAtIndex:0]];
    BOOL isPunct2 = [[NSCharacterSet punctuationCharacterSet] characterIsMember:[(NSString*)obj2 characterAtIndex:0]];
    if (isPunct1 && !isPunct2) {
        return NSOrderedAscending;
    } else if (!isPunct1 && isPunct2) {
        return NSOrderedDescending;
    }
    return [(NSString*)obj1 compare:obj2 options:NSDiacriticInsensitiveSearch|NSCaseInsensitiveSearch];         
}];

要将英文字符放在英文字符之前,用作选项就足够了,不需要花哨的算法。NSDiacriticInsensitiveSearch | NSCaseInsensitiveSearch

如果您需要无块支持 iOS,请尝试sortedArrayUsingSelector.

于 2011-11-23T14:02:55.680 回答
4

通过测试的另一种解决方案,如果字符串是 latin1 可编码的:

  • 测试两个字符串,如果第一个字符是拉丁(又名英文)字符
  • 如果两者都是,则测试两者是否都以字母或数字开头。在这两种情况下,将其保留在compare:
  • 否则,如果一个以数字开头,一个以字母开头,return NSOrderingAscending,如果一个以字母开头,否则NSOrderingDescending

  • 如果两个字符串都不是拉丁语,让我们compare:再决定

  • 如果一个是拉丁语,一个不是,return NSOrderingAscending如果拉丁语是第一个,否则NSOrderingDescending

编码

NSArray *array = [NSArray arrayWithObjects:@"peach",@"apple",@"7",@"banana",@"ananas",@"5", @"papaya",@"4",@"구",@"결",@"1" ,nil];

array = [array sortedArrayUsingComparator:^NSComparisonResult(id obj1, id obj2) {
    NSString *s1 = [obj1 substringToIndex:1];
    NSString *s2 = [obj2 substringToIndex:1];
    BOOL b1 = [s1 canBeConvertedToEncoding:NSISOLatin1StringEncoding];
    BOOL b2 = [s2 canBeConvertedToEncoding:NSISOLatin1StringEncoding];

    if ((b1 == b2) && b1) {//both number or latin char
        NSRange r1 = [s1 rangeOfCharacterFromSet:[NSCharacterSet decimalDigitCharacterSet]];
        NSRange r2 = [s2 rangeOfCharacterFromSet:[NSCharacterSet decimalDigitCharacterSet]];
        if (r1.location == r2.location ) { // either both start with a number or both with a letter
            return [obj1 compare:obj2 options:NSDiacriticInsensitiveSearch|NSCaseInsensitiveSearch]; 
        } else {  // one starts wit a letter, the other with a number
            if ([s1 rangeOfCharacterFromSet:[NSCharacterSet decimalDigitCharacterSet]].location == NSNotFound) {
                return NSOrderedAscending;
            }
            return NSOrderedDescending;
        }
    } else if((b1 == b2) && !b1){ // neither latin char
        return [obj1 compare:obj2 options:NSDiacriticInsensitiveSearch|NSCaseInsensitiveSearch];
    } else { //one is latin char, other not
        if (b1) return NSOrderedAscending;
        return NSOrderedDescending;
    }

}];
for (NSString *s in array) NSLog(@"%@", s);

结果

ananas
apple
banana
papaya
peach
1
4
5
7
구
결
于 2012-04-30T18:13:30.053 回答
3

我不认为你可以在不定义自己的比较函数的情况下进行这种排序。

为此,您可以使用sortedArrayUsingFunction

[array sortedArrayUsingFunction:f context:userContext];

其中 f 定义为:

NSInteger f(id num1, id num2, void *context)
{
   int v1 = [num1 intValue];
   int v2 = [num2 intValue];
   if (...)
     return NSOrderedAscending;
   else if (...)
     return NSOrderedDescending;
   else
     return NSOrderedSame;
}

如果您不想为此创建函数,则可以使用该方法的块版本sortedArrayUsingComparator

[array sortedArrayUsingComparator: ^(id obj1, id obj2) {
                                             return NSOrderedSame;
                                   }];
于 2011-11-23T13:55:11.990 回答
1

基于比较器的排序描述符应该可以解决问题(注意:未经测试)。

NSComparator cmp = ^(id str1, id str2) {

// Make your sorting
    if ( /* str1 before str2 */ )
    return NSOrderedAscending
    else if ( /* str2 after str1 */ )
    return NSOrderedDescending
    else 
    return NSOrderedSame
};

NSSortDescriptor *sd = [NSSortDescriptor sortDescriptorWithKey: sortKey ascending: YES comparator: cmp];

NSArrayController *ac = // ...

[ac setSortDescriptor: sd];

你当然必须定义你自己的排序算法——但是这个例子应该展示如何为数组控制器使用排序描述符。

于 2011-11-23T14:03:40.163 回答
0

缺少一件事来正确回答这个问题:NSNumericSearch

NSArray *assorted = [@"1 2 3 9 ; : 구 , 결 A B C Z ! á" componentsSeparatedByCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
NSArray *sorted = [assorted sortedArrayUsingComparator:^NSComparisonResult(id obj1, id obj2) {
    /* NSOrderedAscending, NSOrderedSame, NSOrderedDescending */
    BOOL isPunct1 = [[NSCharacterSet punctuationCharacterSet] characterIsMember:[(NSString*)obj1 characterAtIndex:0]];
    BOOL isPunct2 = [[NSCharacterSet punctuationCharacterSet] characterIsMember:[(NSString*)obj2 characterAtIndex:0]];
    if (isPunct1 && !isPunct2) {
        return NSOrderedAscending;
    } else if (!isPunct1 && isPunct2) {
        return NSOrderedDescending;
    }
    return [(NSString*)obj1 compare:obj2 options:NSDiacriticInsensitiveSearch|NSCaseInsensitiveSearch|NSNumericSearch]|;         
}];
于 2017-04-26T15:00:10.647 回答