sortUsingSelector使用方法,以及原理

在IOS软件开发中我们需要依据NSArray中对象的成员进行排序,我们知道对象中成员比较多,因此如果我们写了多种方法根据不同的成员进行排序,代码量很大,实现起来不是很方便,因此苹果向我们提供了一种快速排序的方法,叫做

- (void)sortUsingSelector:(SEL)comparator;

本文重视讲诉原理:

我们先来看一个例子:

#import <Foundation/Foundation.h>

//Dog头文件
@interface Dog : NSObject
{
    NSString * _name;
    NSUInteger _age;
}
-(NSComparisonResult)nameCompae:(Dog *)dog;
-(NSComparisonResult)ageCompae:(Dog *)dog;
-(id)initWithName:(NSString *)name andWithAge:(NSUInteger)age;
@property (copy,readwrite,nonatomic) NSString * name;
@property (assign,readwrite,nonatomic)NSUInteger age;
@end

//Dog函数实现体
@implementation Dog

-(NSComparisonResult)nameCompae:(Dog *)dog
{
    if ([_name compare:[dog name]]>0)
    {
        return NSOrderedDescending;
    }
    
    return NSOrderedAscending;
}

-(NSComparisonResult)ageCompae:(Dog *)dog
{
    
    NSLog(@"%ld %ld",_age,[dog age]);
    
    if (_age > [dog age]) {
        
        return NSOrderedAscending;
    }
    return NSOrderedDescending;
}
-(void)setAge:(NSUInteger)age
{
    _age = age;
    NSLog(@"dasdsd");
}

-(id)initWithName:(NSString *)name andWithAge:(NSUInteger)age
{
    if (self = [super init]) {
        _name = name;
        _age = age;
    }
    return self;
}

@end

//main函数
int main(int argc, const char * argv[])
{

    @autoreleasepool {
        
        NSMutableArray * dogArray = [[NSMutableArray alloc]init];
        
        Dog * dog1 = [[Dog alloc]initWithName:@"xiaobao" andWithAge:12];
        Dog * dog2 = [[Dog alloc]initWithName:@"zhansan" andWithAge:8];
        Dog * dog3 = [[Dog alloc]initWithName:@"lisi" andWithAge:16];
        Dog * dog4 = [[Dog alloc]initWithName:@"wangwu" andWithAge:7];
        
        [dogArray addObject:dog1];
        [dogArray addObject:dog2];
        [dogArray addObject:dog3];
        [dogArray addObject:dog4];
        
        for (Dog * temp in dogArray) {
            NSLog(@"%@ %ld",temp.name,temp.age);
        }
        
        [dogArray sortUsingSelector:@selector(nameCompae:)];
        
        
        for (Dog * temp in dogArray)
        {
            NSLog(@"%@ %ld",temp.name,temp.age);
        }
        
        [dogArray sortUsingSelector:@selector(ageCompae:)];
        
        
        for (Dog * temp in dogArray)
        {
            NSLog(@"%@ %ld",temp.name,temp.age);
        }
    }
    return 0;
}

在上面的这个例子中,我将小狗对象存入dogArray中,而该对象中又有两个成员变量,一个是狗的名字,另一个是狗的年龄,我们通过这两个属性对这个dogArray中得对象进行排序,在测试中我发现在调用sortUsingSelector函数时,选择器(即ageCompae:和nameCompae)是被多次调用的,因此我猜测,该函数的伪代码如下:

- (void)mysortUsingSelector:(SEL)comparator:①

{

int temp = 0;

for (int i=0; i<[self count]-1; i++)②

{   temp = i;

for (int j=i+1; j<[self count]; j++)

{

if ([[self objectAtIndex:temp] comparator:[self objectAtIndex:j]]==NSOrderedDescending)③

{

temp = j;

}

}

if (i!=temp)

{

[dogArray exchangeObjectAtIndex:i withObjectAtIndex:temp];

}

}

}

上述代码中又三点需要理解,只要理解了这个伪代码,你就知道sortUsingSelector这个函数是怎么工作的了.

①:(SEL)comparator:这个东西其实就是模仿C语言中的函数指针,我们想要使用什么函数对NSArray的对象只要把函数入口地址传给它就好了,例如上述代码中想使用dog名字进行排序,只要把nameCompae:传给它就好了,而想要按照年龄排序就传ageCompare:如果你理解函数指针,这个对你就很简单了,呵呵

②:这个self其实就是数组的地址,对应与上面代码中的dogArray

③:使用传递过来的函数进行比较,也就是C语言中函数指针的回调函数,上述代码中有一点很重要,也就是

if ([[self objectAtIndex:temp] comparator:[self objectAtIndex:j]]==NSOrderedDescending)这有在这个语句为真的时候才会进行交换,我测试多次sortUsingSelector这个函数在回调(SEL)comparator:函数时,只有在(SEL)comparator:函数返回值为NSOrderedDescending:才会进行排序,对于其它值则不会进行如何处理,这也很好的解释了我上面的if语句的书写方式是正确的.

总结:相信到了这里你一定明白了sortUsingSelector的工作方式了,如果不明白建议看看关于函数指针的视频或者书籍,推荐在51CTO上面找找;

时间: 2024-10-15 05:53:03

sortUsingSelector使用方法,以及原理的相关文章

Atitit.编程语言原理---方法重载的实现与设计&#160;调用方法的原理

Atitit.编程语言原理---方法重载的实现与设计 调用方法的原理 1. 重载包括:普通方法的重载和构造方法的重载 1 1.1. 横向重载”和“纵向重载”1 1.2. 方法签名通过  方法名称,参数数量,参数类型+返回类型 参数顺序(命名参数)来组成1 1.3. 重载(重新载选方法流程,如下图所示2 2. 重载的实现方法::argus参数 vs  默认值.可选参数法 可选参数3 2.1. __call()函数是php类的默认魔法函数 3 2.2. Java通过反射调用方法重载 执行方法4 2.

JavaScript中Object.prototype.toString方法的原理

在JavaScript中,想要判断某个对象值属于哪种内置类型,最靠谱的做法就是通过Object.prototype.toString方法. ? 1 2 var arr = []; console.log(Object.prototype.toString.call(arr)) //"[object Array]" 本文要讲的就是,toString方法是如何做到这一点的,原理是什么. ECMAScript 3 在ES3中,Object.prototype.toString方法的规范如下:

css清除浮动各方法与原理

说到清除浮动的方法,我想网络上应该有不下7,8的方法,介绍这些方法之前,想下为什么清除浮动? 再次回到float这个属性,浮动元素(floats)会被移出文档流,不会影响到块状盒子的布局而只会影响内联盒子(通常是文本)的排列. 这就产生了一个问题:浮动元素所在父元素不会自动伸长以便闭合浮动元素,任其“红杏出墙”,这显然不符合我们的“伦理观”,所以要“抓回来”施以家法:那么如何抓呢? 提起css,提起(x)html,首先进入脑海的是楚楚动人的页面吗?不,首先应该是这楚楚动人的页面的背后的“楚楚冻人

iOS端JSON转Model链式编程框架SuperKVC使用方法与原理

背景 在client编程中.字典转模型是一个极为常见的问题,苹果提供了KVC来实现NSDictionary到Model的注入,可是KVC仅仅能进行单层浅注入.且无法处理类型转换.key与属性名不正确应.深度注入等问题,笔者从Masonry得到启示,开发了一个通过链式配置注入器实现深度注入.类型转换.key-属性名映射等功能的轻量级注入框架SuperKVC.眼下已经开源到GitHub,点击这里前往.欢迎Star和Fork.欢迎和我一起完好这个框架! 本文将从应用和原理两个角度介绍SuperKVC

网络转载:局域网安全:解决ARP攻击的方法和原理

局域网安全:解决ARP攻击的方法和原理 IT世界网2006-01-26 10:17 [故障原因] 局域网内有人使用ARP欺骗的木马程序(比如:传奇盗号的软件,某些传奇外挂中也被恶意加载了此程序). [故障原理] 要了解故障原理,我们先来了解一下ARP协议. 在局域网中,通过ARP协议来完成IP地址转换为第二层物理地址(即MAC地址)的.ARP协议对网络安全具有重要的意义.通过伪造IP地址和MAC地址实现ARP欺骗,能够在网络中产生大量的ARP通信量使网络阻塞. ARP协议是“Address Re

深入理解前端跨域方法和原理

前言 受浏览器同源策略的限制,本域的js不能操作其他域的页面对象(比如DOM).但在安全限制的同时也给注入iframe或是ajax应用上带来了不少麻烦.所以我们要通过一些方法使本域的js能够操作其他域的页面对象或者使其他域的js能操作本域的页面对象(iframe之间). 这里需要明确的一点是:所谓的域跟js的存放服务器没有关系,比如baidu.com的页面加载了google.com的js,那么此js的所在域是baidu.com而不是google.com.也就是说,此时该js能操作baidu.co

Android ListView实现不同item的方法和原理分析

ListView实现不同item的方法和原理分析 一问题抛出Listview是android里面的重要组件,用来显示一个竖向列表,这个没有什么问题:但是有个时候列表里面的item不是一样的,如下图,列表里面应该有3种类型的item  1. 头像在左边的气泡Item ,比如”今天下午我就不出来了,...”2. 头像在右边的气泡Item,比如”那就等着我发你好吧”3. 单张图片显示圆角图片item几种Item的风格是完全不同的,那么怎么实现呢? 二实现方法实现的方法我这里可以列举出两种1. 每个It

android中获取root权限的方法以及原理(转)

一. 概述 本文介绍了android中获取root权限的方法以及原理,让大家对android 玩家中常说的“越狱”有一个更深层次的认识. 二. Root 的介绍 1. Root 的目的 可以让我们拥有掌控手机系统的权限,比如删除一些system/app下面的无用软件,更换开关机铃声和动画,拦截状态栏弹出的广告等. 2. Root的原理介绍 谷歌的android系统管理员用户就叫做root,该帐户拥有整个系统至高无上的权利,它可以访问和修改你手机几乎所有的文件,只有root才具备最高级别的管理权限

jquery ready方法实现原理 内部原理

今天闲来无事研究研究jquery.ready()的内部实现,看JQ的源码一头雾水,由于自己很菜了,于是翻了翻牛人的播客,讲述详细,收获颇多. 先普及一下jquery.ready()和window.onload,window.onload事件是在页面所有的资源都加载完毕后触发的. 如果页面上有大图片等资源响应缓慢, 会导致window.onload事件迟迟无法触发.所以出现了DOM Ready事件. 此事件在DOM文档结构准备完毕后触发, 即在资源加载前触发. 我的ready方法写了2版,借鉴了不