在做项目的时候,总遇到排序问题,英文排序是很简单的,直接使用compare方法就可以实现了,但是一旦遇到中文,就显得麻烦了。

iOS和java不同,java能进行中文字符串的比较,而iOS却没有这个功能。于是乎,iOS中的中文排序就要自己想办法了。相信很多人想到的方法就是获得中文字符串的拼音,再进行排序了。

之前也找过中文排序的相关文章,想必*出名的就是ChineseString.h和pinyin.c文件(或者其变体)了,对于这个pinyin.c文件里密密麻麻的我表示我没认真看懂~~,自己想着就是中文拼音的对应表之类的。但是我使用这两个文件会出现问题的,排序是有进行排序了,但是并不是按照严格的全拼音排序,它只是对中文字符串首字母进行了排序,之后的就没有排了。

在某些场景中,这的确就是足够了,但是现在我需要的是全拼音排序,就是要对中文字符串整个拼音排序。在对着pinyin.c一大串没看懂的代码面前,想着还是自己想办法弄一个吧。在这个过程中,获得拼音是*个难关,之后的通过拼音进行排序说到底就是进行英文字符串的比较。幸好,*终还是在网上找到获得中文字符串拼音的方法。

首先总体思路是这样的:

1. 获得拼音进行比较排序是必须的了,排序就意味着存在数组,但是原本的中文字符串该是如何存储的呢,应该怎样存在数组中呢。我自个儿想了三种方式,一是简单直接的中文字符串数组。接着想着有时候需要排序的中文字符串可能还附加其他的属性,于是有了第二中方式,就是使用字典(NSDictionary)存储需要排序的中文字符串以及其他属性。一个例子就是现在需要对通讯录的人名排序,而通讯录每个人除了人名,至少还包含电话号码,那么就要确保按照人名排序完成后,电话号码也相应的进行了排序了,那么在这个例子中使用第二种方式就是通讯录就是一个数组,而数组的元素就是字典,字典包含了两个属性,一个是人名,另一个是电话号码。而第三种方式是从第二种方式延伸出来的,还是以通讯录排序为例子,我可以把通讯录中每个人都看成一个类的实例(对象),这样人名和电话号码就是类中的属性,这样的话,通讯录是一个数组,而数组的元素不再是字典,而是一个个类的实例了。

*种方式数组初始化Demo:

NSArray *a1 = @[@”张三”, @”李三”, @”li三”, @”qeri三”, @”rei三”, @”lbcx三”, @”kjh三”, @”照三”, @”于三”, @”破三”, @”梁三”];
第二种方式数组初始化Demo:

NSArray *a2 = @[
@{@”name”:@”张三”, @”num”:@”1″},
@{@”name”:@”李三”, @”num”:@”2″},
@{@”name”:@”li三”, @”num”:@”3″},
@{@”name”:@”qeri三”, @”num”:@”4″},
@{@”name”:@”rei三”, @”num”:@”5″},
@{@”name”:@”lbcx三”, @”num”:@”6″},
@{@”name”:@”kjh三”, @”num”:@”7″},
@{@”name”:@”照三”, @”num”:@”8″},
@{@”name”:@”于三”, @”num”:@”9″},
@{@”name”:@”破三”, @”num”:@”10″},
@{@”name”:@”梁三”, @”num”:@”11″}];

第三种方式数组初始化Demo:(这里说明一下,JHChineseInfo是只有一个属性(JHChineseString)的类,如果要使用这种方式进行排序,自定义的类必须继承JHChineseInfo,就如下面的SubChineseInfo,这是这种方式的一点限制,如有大神路过,请给点改进的意见,谢谢~)

JHChineseInfo *ci1 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci2 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci3 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci4 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci5 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci6 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci7 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci8 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci9 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci10 = [[JHChineseInfo alloc] init];
JHChineseInfo *ci11 = [[JHChineseInfo alloc] init];
ci1.JHChineseString = @”张三”;
ci2.JHChineseString = @”李三”;
ci3.JHChineseString = @”li三”;
ci4.JHChineseString = @”qeri三”;
ci5.JHChineseString = @”rei三”;
ci6.JHChineseString = @”lbcx三”;
ci7.JHChineseString = @”kjh三”;
ci8.JHChineseString = @”照三”;
ci9.JHChineseString = @”于三”;
ci10.JHChineseString = @”破三”;
ci11.JHChineseString = @”梁三”;
NSArray *a3 = @[ci1, ci2, ci3, ci4, ci5, ci6, ci7, ci8, ci9, ci10, ci11];
或者:
SubChineseInfo *sci1 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci2 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci3 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci4 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci5 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci6 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci7 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci8 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci9 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci10 = [[SubChineseInfo alloc] init];
SubChineseInfo *sci11 = [[SubChineseInfo alloc] init];
sci1.JHChineseString = @”张三”;
sci1.num = @”1″;
sci2.JHChineseString = @”李三”;
sci2.num = @”2″;
sci3.JHChineseString = @”li三”;
sci3.num = @”3″;
sci4.JHChineseString = @”qeri三”;
sci4.num = @”4″;
sci5.JHChineseString = @”rei三”;
sci5.num = @”5″;
sci6.JHChineseString = @”lbcx三”;
sci6.num = @”6″;
sci7.JHChineseString = @”kjh三”;
sci7.num = @”7″;
sci8.JHChineseString = @”照三”;
sci8.num = @”8″;
sci9.JHChineseString = @”于三”;
sci9.num = @”9″;
sci10.JHChineseString = @”破三”;
sci10.num = @”10″;
sci11.JHChineseString = @”梁三”;
sci11.num = @”11″;
NSArray *a4 = @[sci1, sci2, sci3, sci4, sci5, sci6, sci7, sci8, sci9, sci10, sci11];

2. 获得中文字符串的拼音
函数- (NSString*) chineseStringTransformPinyin: (NSString*)chineseString

// 拼音字段
NSMutableString *tempNamePinyin = [chineseString mutableCopy];
CFStringTransform((__bridge CFMutableStringRef)tempNamePinyin, NULL, kCFStringTransformMandarinLatin, NO);
CFStringTransform((__bridge CFMutableStringRef)tempNamePinyin, NULL, kCFStringTransformStripDiacritics, NO);
这个方法就是从万能的网上寻找回来的,具体说明的话当时看看了,没记录网址就忘了。现在自己产生了一个问题,不知这个方法在效率上是否*优,还有没有更好的方法。在此再请路过的大神知道的指点一下~~

3. 这一步就开始进行中文排序了,上述的三种方式分别对应了三个方法:

1)- (NSArray*) chineseSortWithStringArray: (NSArray*)stringArray;

stringArray必定要是字符串数组,就是数组的元素就是需要排序的中文字符串

2)- (NSArray*) chineseSortWithDictionaryArray: (NSArray*)dictionaryArray andFieldKey: (NSString*)fieldKey

dictionaryArray是字典数组,就是数组的元素是字典,其中key值为fieldKey对应的值就是需要排序的中文字符串

3)- (NSArray*) chineseSortWithObjectArray: (NSArray*)objectArray

objectArray是对象数组,就是数组的元素一定要是JHChineseInfo或者其子类(一般如需自定义就自定义一个子类出来,把需要排序的中文字符串字段赋给JHChineseString属性即可)

排序关键代码(其实就是通过pinyin字段进行英文字符串比较获得排序结果):

[tempArray sortUsingComparator:^NSComparisonResult(id obj1, id obj2) {
return [[obj1 objectForKey:@”pinyin”] compare:[obj2 objectForKey:@”pinyin”]];
}];
*终返回结果。

我把获得拼音以及三种方式的函数独立出来了,在使用的时候只需要导入JHChineseSort.h、JHChineseSort.m、JHChineseInfo.h以及JHChineseInfo.m四个文件即可调用。

至此,完结~

如有问题获建议,可一起讨论,共同学习~