[iOS] 浅析排序规则描述类: NSSortDescriptor

NSSortDescriptor是用来指定排序规则, 对集合等进行排序时指定结果的排序规则;
他可以对一个类的某个属性(下文中方法中的key参数)指定排序规则, 也可以对一个字符串集合进行指定排序规则, 这时, 只需要把参数key赋值为nil即可.
iOS中的集合都有相应的扩展方法来使用NSSortDescriptor:

  • NSSet
-(NSArray<ObjectType> *)sortedArrayUsingDescriptors:(NSArray<NSSortDescriptor *> *)sortDescriptors
  • NSArray
-(NSArray<ObjectType> *)sortedArrayUsingDescriptors:(NSArray<NSSortDescriptor *> *)sortDescriptors;
  • NSMutableArray
-(void)sortUsingDescriptors:(NSArray<NSSortDescriptor *> *)sortDescriptors;
  • NSOrderedSet
-(NSArray<ObjectType> *)sortedArrayUsingDescriptors:(NSArray<NSSortDescriptor *> *)sortDescriptors
  • NSMutableOrderedSet
-(void)sortUsingDescriptors:(NSArray<NSSortDescriptor *> *)sortDescriptors

上面的参数都是包含NSSortDescriptor的数组, 意思是说可以同时指定多条规则来进行排序, 其优先级取决于在数组中的先后顺序;

初始化方法

初始化方法有以下几种:

+ (instancetype)sortDescriptorWithKey:(nullable NSString *)key ascending:(BOOL)ascending NS_AVAILABLE(10_6, 4_0);
+ (instancetype)sortDescriptorWithKey:(nullable NSString *)key ascending:(BOOL)ascending selector:(nullable SEL)selector NS_AVAILABLE(10_6, 4_0);

// keys may be key paths
- (instancetype)initWithKey:(nullable NSString *)key ascending:(BOOL)ascending;
- (instancetype)initWithKey:(nullable NSString *)key ascending:(BOOL)ascending selector:(nullable SEL)selector;

+ (instancetype)sortDescriptorWithKey:(nullable NSString *)key ascending:(BOOL)ascending comparator:(NSComparator)cmptr NS_AVAILABLE(10_6, 4_0);
- (instancetype)initWithKey:(nullable NSString *)key ascending:(BOOL)ascending comparator:(NSComparator)cmptr NS_AVAILABLE(10_6, 4_0);

下面通过一个示例来说明各个方法的使用规则:

创建假数据

这里定义了一个简单model:

// People.h
@interface People : NSObject

@property (nonatomic, copy) NSString *name;
@property (nonatomic, assign) BOOL sex;
@property (nonatomic, assign) double height;
@property (nonatomic, assign) int age;
@end

// People.m
@implementation People
@end

生成假数据:

 NSArray *names = @[@"夏侯惇", @"貂蝉", @"诸葛亮", @"张三", @"李四", @"流火绯瞳", @"流火", @"李白", @"张飞", @"韩信", @"范冰冰", @"赵丽颖"];
 NSArray *ages = @[@32, @32, @45, @32, @32, @27, @15, @67, @55, @34, @44, @30];
 NSArray *heights = @[@170, @163, @180, @165, @163, @176, @174, @183, @186, @178, @167, @160];
    
    NSMutableArray *peoples = [NSMutableArray arrayWithCapacity:names.count];
    for (int i = 0; i<names.count; i++) {
        
        People *pe = [[People alloc]init];
        pe.name = names[i];
        pe.age = [ages[i] intValue];
        pe.height = [heights[i] doubleValue];
        [peoples addObject:pe];
    }

以下示例代码, 都是对这个peoples数组进行操作

+ (instancetype)sortDescriptorWithKey:(nullable NSString *)key ascending:(BOOL)ascending
- (instancetype)initWithKey:(nullable NSString *)key ascending:(BOOL)ascending

参数
key : 排序key, 某个对象的属性名称; 如果对字符串进行排序, 则传nil
ascending : 是否升序, YES-升序, NO-降序

示例代码:

NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:YES];
   
 [peoples sortUsingDescriptors:@[sort]];
    
    // 输出排序结果
    for (People *people in peoples) {
        NSLog(@"age: %d,height: %f name: %@", people.age,people.height, people.name);
    }

输出:

2017-06-30 16:25:39.628 NSPredicateTest[10794:419605] age: 15,height: 174.0 name: 流火
2017-06-30 16:25:39.629 NSPredicateTest[10794:419605] age: 27,height: 176.0 name: 流火绯瞳
2017-06-30 16:25:39.630 NSPredicateTest[10794:419605] age: 30,height: 160.0 name: 赵丽颖
2017-06-30 16:25:39.630 NSPredicateTest[10794:419605] age: 32,height: 170.0 name: 夏侯惇
2017-06-30 16:25:39.631 NSPredicateTest[10794:419605] age: 32,height: 163.0 name: 貂蝉
2017-06-30 16:25:39.631 NSPredicateTest[10794:419605] age: 32,height: 165.0 name: 张三
2017-06-30 16:25:39.631 NSPredicateTest[10794:419605] age: 32,height: 163.0 name: 李四
2017-06-30 16:25:39.631 NSPredicateTest[10794:419605] age: 34,height: 178.0 name: 韩信
2017-06-30 16:25:39.632 NSPredicateTest[10794:419605] age: 44,height: 167.0 name: 范冰冰
2017-06-30 16:25:39.632 NSPredicateTest[10794:419605] age: 45,height: 180.0 name: 诸葛亮
2017-06-30 16:25:39.632 NSPredicateTest[10794:419605] age: 55,height: 186.0 name: 张飞
2017-06-30 16:25:39.632 NSPredicateTest[10794:419605] age: 67,height: 183.0 name: 李白

也可以同时指定多个规则: 按年龄升序排列, 相同的再按身高降序排列:

NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:YES];
NSSortDescriptor *sort1 = [NSSortDescriptor sortDescriptorWithKey:@"height" ascending:NO];
[peoples sortUsingDescriptors:@[sort, sort1]];
    
    // 输出排序结果
    for (People *people in peoples) {
        NSLog(@"age: %d,height: %.1f name: %@", people.age,people.height, people.name);
    }

输出:

2017-06-30 16:27:27.489 NSPredicateTest[10856:421571] age: 15,height: 174.0 name: 流火
2017-06-30 16:27:27.490 NSPredicateTest[10856:421571] age: 27,height: 176.0 name: 流火绯瞳
2017-06-30 16:27:27.490 NSPredicateTest[10856:421571] age: 30,height: 160.0 name: 赵丽颖
2017-06-30 16:27:27.490 NSPredicateTest[10856:421571] age: 32,height: 170.0 name: 夏侯惇
2017-06-30 16:27:27.490 NSPredicateTest[10856:421571] age: 32,height: 165.0 name: 张三
2017-06-30 16:27:27.490 NSPredicateTest[10856:421571] age: 32,height: 163.0 name: 貂蝉
2017-06-30 16:27:27.490 NSPredicateTest[10856:421571] age: 32,height: 163.0 name: 李四
2017-06-30 16:27:27.491 NSPredicateTest[10856:421571] age: 34,height: 178.0 name: 韩信
2017-06-30 16:27:27.491 NSPredicateTest[10856:421571] age: 44,height: 167.0 name: 范冰冰
2017-06-30 16:27:27.491 NSPredicateTest[10856:421571] age: 45,height: 180.0 name: 诸葛亮
2017-06-30 16:27:27.491 NSPredicateTest[10856:421571] age: 55,height: 186.0 name: 张飞
2017-06-30 16:27:27.491 NSPredicateTest[10856:421571] age: 67,height: 183.0 name: 李白

这次主要是看年龄为32的排列顺序;

+ (instancetype)sortDescriptorWithKey:(nullable NSString *)key ascending:(BOOL)ascending selector:(nullable SEL)selector
- (instancetype)initWithKey:(nullable NSString *)key ascending:(BOOL)ascending selector:(nullable SEL)selector

参数:
key : 排序key, 某个对象的属性名称
ascending : 是否升序, YES-升序, NO-降序
selector : 自定义排序规则, 如果需要自己定义排序规则, 可传递此方法, 这个使用相对比较复杂; 如果待比较的属性是字符串(NSString)类型, 可使用其默认的方法: localizedStandardCompare:
即:

NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"name" ascending:YES selector:@selector(localizedStandardCompare:)];

如果是其他的类型, 比如int, double等, 就需要对其类别(基本类型要对NSNumber)进行扩展;

假设还按年龄进行排序, 只不过我想修改其排序方法: ascending 为YES时, 让其降序排列(本应该是升序排列), 可以这样做:

首先, 对NSNumber类型写一个类目:

// NSNumber+mySort.h
@interface NSNumber (mySort)

- (NSComparisonResult)mySort:(NSNumber *)num;
@end

// NSNumber+mySort.m
@implementation NSNumber (mySort)
- (NSComparisonResult)mySort:(NSNumber *)num {
    
    if (self == num) {
        return NSOrderedSame;
    } else if (self > num) {
        // 当自身大于num时, 本应该返回 NSOrderedDescending , 这里反转其结果, 使返回 NSOrderedAscending
        return NSOrderedAscending;
    }else {
        return NSOrderedDescending;
    }
}
@end

然后使用:

NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:YES selector:@selector(mySort:)];
    
    [peoples sortUsingDescriptors:@[sort]];
    
    // 输出排序结果
    for (People *people in peoples) {
        NSLog(@"age: %d,height: %.1f name: %@", people.age,people.height, people.name);
    }

这时输出的结果就和上面相反了:

2017-06-30 16:47:44.192 NSPredicateTest[11287:445115] age: 67,height: 183.0 name: 李白
2017-06-30 16:47:44.192 NSPredicateTest[11287:445115] age: 55,height: 186.0 name: 张飞
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 45,height: 180.0 name: 诸葛亮
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 44,height: 167.0 name: 范冰冰
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 34,height: 178.0 name: 韩信
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 32,height: 170.0 name: 夏侯惇
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 32,height: 163.0 name: 貂蝉
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 32,height: 165.0 name: 张三
2017-06-30 16:47:44.193 NSPredicateTest[11287:445115] age: 32,height: 163.0 name: 李四
2017-06-30 16:47:44.194 NSPredicateTest[11287:445115] age: 30,height: 160.0 name: 赵丽颖
2017-06-30 16:47:44.194 NSPredicateTest[11287:445115] age: 27,height: 176.0 name: 流火绯瞳
2017-06-30 16:47:44.194 NSPredicateTest[11287:445115] age: 15,height: 174.0 name: 流火

如果是其他类型的属性, 可以分别进行扩展, 自定义排序规则, 需要注意的是, 自定义方法的返回值一定要是NSComparisonResult.

+ (instancetype)sortDescriptorWithKey:(nullable NSString *)key ascending:(BOOL)ascending comparator:(NSComparator)cmptr
- (instancetype)initWithKey:(nullable NSString *)key ascending:(BOOL)ascending comparator:(NSComparator)cmptr

参数:
key : 排序key, 某个对象的属性名称
ascending : 是否升序, YES-升序, NO-降序
cmptr: 一个block, 可以在其中指定比较规则

这里的初始化方法, 和上面的方法功能一样, 可以自己提供一个排序规则, 只不过方式不同; 这个相对于上面的方法, 在自定义排序方法不需要新建扩展, 相对方便一些:

示例:

 NSSortDescriptor *sort = [NSSortDescriptor sortDescriptorWithKey:@"age" ascending:YES comparator:^NSComparisonResult(id  _Nonnull obj1, id  _Nonnull obj2) {
        
        if (obj1 == obj2) {
            return NSOrderedSame;
        } else if (obj1 > obj2){
            // 当obj1大于obj2时, 本应该返回 NSOrderedDescending , 这里反转其结果, 使返回 NSOrderedAscending
            return NSOrderedAscending;
        } else {
            return NSOrderedDescending;
        }
    }];
    
    [peoples sortUsingDescriptors:@[sort]];
    
    // 输出排序结果
    for (People *people in peoples) {
        NSLog(@"age: %d,height: %.1f name: %@", people.age,people.height, people.name);
    }

这样的结果和上面是一样的:

2017-06-30 16:55:22.049 NSPredicateTest[11432:452683] age: 67,height: 183.0 name: 李白
2017-06-30 16:55:22.050 NSPredicateTest[11432:452683] age: 55,height: 186.0 name: 张飞
2017-06-30 16:55:22.050 NSPredicateTest[11432:452683] age: 45,height: 180.0 name: 诸葛亮
2017-06-30 16:55:22.050 NSPredicateTest[11432:452683] age: 44,height: 167.0 name: 范冰冰
2017-06-30 16:55:22.050 NSPredicateTest[11432:452683] age: 34,height: 178.0 name: 韩信
2017-06-30 16:55:22.050 NSPredicateTest[11432:452683] age: 32,height: 170.0 name: 夏侯惇
2017-06-30 16:55:22.051 NSPredicateTest[11432:452683] age: 32,height: 163.0 name: 貂蝉
2017-06-30 16:55:22.051 NSPredicateTest[11432:452683] age: 32,height: 165.0 name: 张三
2017-06-30 16:55:22.051 NSPredicateTest[11432:452683] age: 32,height: 163.0 name: 李四
2017-06-30 16:55:22.051 NSPredicateTest[11432:452683] age: 30,height: 160.0 name: 赵丽颖
2017-06-30 16:55:22.051 NSPredicateTest[11432:452683] age: 27,height: 176.0 name: 流火绯瞳
2017-06-30 16:55:22.051 NSPredicateTest[11432:452683] age: 15,height: 174.0 name: 流火

以上便是其初始化方法的使用规则.

如果想得到相反的结果, 达到和上面相同的效果, 还有一个方法, 就是自定义MYSortDescriptor继承自NSSortDescriptor, 重写方法:

- (NSComparisonResult)compareObject:(id)object1 toObject:(id)object2;    // primitive - override this method if you want to perform comparisons differently (not key based for example)

例如:

// MYSortDescriptor.h
@interface MYSortDescriptor : NSSortDescriptor

@end

// MYSortDescriptor.m
@implementation MYSortDescriptor

- (NSComparisonResult)compareObject:(id)object1 toObject:(id)object2 {
    // 这里反转其比较顺序
    return [super compareObject:object2 toObject:object1];
}
@end

使用:

MYSortDescriptor *sort = [MYSortDescriptor sortDescriptorWithKey:@"age" ascending:YES];

    [peoples sortUsingDescriptors:@[sort]];
    
    // 输出排序结果
    for (People *people in peoples) {
        NSLog(@"age: %d,height: %.1f name: %@", people.age,people.height, people.name);
    }

其结果, 和上面一样, 不过这个方法不常用, 上面的方法已经能够满足日常所需.

以上便是本人对 NSSortDescriptor 的学习总结, 如有不到之处, 还请指正!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,496评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,407评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,632评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,180评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,198评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,165评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,052评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,910评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,324评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,542评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,711评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,424评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,017评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,668评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,823评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,722评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,611评论 2 353

推荐阅读更多精彩内容