美文网首页
iOS 通讯录中文排序、全拼音排序

iOS 通讯录中文排序、全拼音排序

作者: 超级卡布达 | 来源:发表于2021-04-19 15:28 被阅读0次

    示例Demo在GitHub:https://github.com/976971956/ChineseSortDemo
    在做项目的时候,总遇到排序问题,英文排序是很简单的,直接使用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];

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

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

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

    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。

    相关文章

      网友评论

          本文标题:iOS 通讯录中文排序、全拼音排序

          本文链接:https://www.haomeiwen.com/subject/cybilltx.html