在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上面找找;
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。