在做项目的时候,总遇到排序问题,英文排序是很简单的,直接使用compare方法就可以实现了,但是一旦遇到中文,就显得麻烦了。
iOS和java不同,java能进行中文字符串的比较,而iOS却没有这个功能。于是乎,iOS中的中文排序就要自己想办法了。相信很多人想到的方法就是获得中文字符串的拼音,再进行排序了。
之前也找过中文排序的相关文章,想必最出名的就是ChineseString.h和pinyin.c文件(或者其变体)了,对于这个pinyin.c文件里密密麻麻的我表示我没认真看懂~~,自己想着就是中文拼音的对应表之类的。但是我使用这两个文件会出现问题的,排序是有进行排序了,但是并不是按照严格的全拼音排序,它只是对中文字符串首字母进行了排序,之后的就没有排了。
在某些场景中,这的确就是足够了,但是现在我需要的是全拼音排序,就是要对中文字符串整个拼音排序。在对着pinyin.c一大串没看懂的代码面前,想着还是自己想办法弄一个吧。在这个过程中,获得拼音是第一个难关,之后的通过拼音进行排序说到底就是进行英文字符串的比较。幸好,最终还是在网上找到获得中文字符串拼音的方法。
首先总体思路是这样的:
- 获得拼音进行比较排序是必须的了,排序就意味着存在数组,但是原本的中文字符串该是如何存储的呢,应该怎样存在数组中呢。我自个儿想了三种方式,一是简单直接的中文字符串数组。接着想着有时候需要排序的中文字符串可能还附加其他的属性,于是有了第二中方式,就是使用字典(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"]]; }];
最终返回结果。
4.使用规则
可自定义模型继承JHChineseInfo,可先将字典转模型,在循环给name,num赋值。也可将JHChineseInfo中name改名字,但JHChineseSort中所有name都要改。
我把获得拼音以及三种方式的函数独立出来了,在使用的时候只需要导入JHChineseSort.h、JHChineseSort.m、JHChineseInfo.h、JHChineseInfo.m以及 笔画文件 KLChineseCharactersTool.h、KLChineseCharactersTool.m、Chinese.json、byteNum.json。