IOS将一组包含中文的数据按照#ABC...Z?分组
Posted 汪小饭
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了IOS将一组包含中文的数据按照#ABC...Z?分组相关的知识,希望对你有一定的参考价值。
上一篇文章【ios】模仿windowsphone列表索引控件YFMetroListBox里面
我们一步步的实现了WindowsPhone风格的索引。
但是有没有发现,如果你要实现按照字母排序,你还得自己填入这些数据,而不能够让其自动归类。
因此我们这篇文章来说说如何将一个数组进行排序。
标题中的#代表数字、✿代表除了数字、中英文外的其他符号。
1.需求 :将字符串分类成数字、中英文首字母、其他符号三类。
数字、符号按照首字母排序,每一个字母分类中,英文始终在中文前面
这里面主要涉及到了对中文英文进行排序。
2.弯路 :
1.网络上有一个pinyin.h. 能够获取到中文的首字母,但是仅仅获取到每个字的首字母对我们来说
还是没有多大用途。 两个首字母相同的中文不知道到底哪个在前,哪个在后。
2.仅仅是一个中文,并不能够判断其可能是多音字中的哪一个(当然本篇随笔也没有涉及到多音字的处理)
3.储备 :
所以最好的方法就是将中文转换为拼音字符串,然后对字符串排序。
将中文转换为拼音,IOS已经给出了方法了(效率什么的暂时不管)
-(NSString *)transformToPinyin{
NSMutableString *mutableString = [NSMutableString stringWithString:self]; //self指的是该汉字字符串
CFStringTransform((CFMutableStringRef)mutableString, NULL, kCFStringTransformToLatin, false);//带有声调的拼音
CFStringTransform((CFMutableStringRef)mutableString, NULL, kCFStringTransformStripDiacritics, false);//去除声调
return mutableString;
}
利用正则表达式判断字符串的第一位是否是数字、中文、英文
-(BOOL)startWith{
NSString *firsetStr = [self substringToIndex:1];
NSString *regex = @"[\\u4e00-\\u9fa5]";//中文 @"[a-zA-Z\\\\s]"为英文 @"[0-9]"为数字
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF MATCHES %@",regex];
return [predicate evaluateWithObject:firsetStr];
}
我们分类出来的每一类都是一个数组
可以使用自定义函数sortedArrayUsingFunction进行排序。
[groupedArray addObject:[numberArray sortedArrayUsingFunction:normalSort context:NULL]];
可以通过localizedCompare此方法来进行比较两个字符串
NSInteger normalSort(id string1, id string2, void *context){ NSString *str1 = (NSString *)string1; NSString *str2 = (NSString *)string2; NSComparisonResult result = [str1 localizedCompare:(NSString*)str2];return result; }
4.解决:
现在我们来实现将他们组装成我们在继承UITableView的YFMetroListBox中需要使用的数据。以下均通过静态方法(类方法)获取。
1.IndexArray; //HeaderView标题数组
- (NSString *)getFirstLetter{
if([self startWithNumber]){
return kNumberSign; //#数字
}else if([self startWithChinese] || [self startWithEnglish]){
NSString *string = [self changeToEnglishOrChinese];
NSString *firstLetter = [string hasPrefix:kDistinguishString] ? [string substringWithRange: NSMakeRange(kDisti nguishString.length, 1)] : [string substringToIndex:1];//取得第一个字母
return firstLetter;
}else{
return kSymbolSign; //✿其他
}
}
-(NSString *)changeToEnglishOrChinese{
NSString *alphabet;
if([self startWithChinese]){//1.中文
alphabet= [self transformToPinyin];
}else{//2.英文 增加@"0" kDistinguishString 保证比较的时候比中文首字母小
alphabet = [kDistinguishString stringByAppendingString:self]; }
return alphabet; }
如何保证转换后拼音的拼音与英文比较 英文始终在中文前面?
我是想法是在拼音前加上一小于@"a"的字符: @"0";
在取得标题首字母数组后 就可以和整体(#ABC...Z✿全部)来进行比较,看存在哪些就可以了
2.groupedArray; //每个分类类别(#.A.B.C...Z.✿)组成的数组
首先通过正则表达式将传入的array分为3大类,数字、中英文(由于英文做了特殊处理,所以可以一起比较了)、其他三大类。
之后对每一类从小到大进行排序。
//给中英文排序
NSArray *indexArray = [self getIndexArray:alphabetArray];//取得IndexArray标题数组 NSArray *headerArray = [self getFirstLetterArray:alphabetArray]; //取得中英文首字母数组
for (int i = 0; i < indexArray.count; i++) {
NSMutableArray *alphaArray = [NSMutableArray array];
for (int j = 0; j < alphabetArray.count; j++) {
NSString *headString = headerArray[j];
if([[headString uppercaseString] isEqualToString:indexArray[i]]){
[alphaArray addObject:alphabetArray[j]];
}
}
[groupedArray addObject:[alphaArray sortedArrayUsingFunction:cnAndEnSort context:NULL]]; //对每一个数组进行排序
}
3.GroupedDictionaryArray: //通过上面两部,取得了两组数据,现在可以以字典的形式组装(包含两类)
1.类似@[{@"indexKey":@"A",@"arrayKey":@[@"abandon",@"啊哈哈",@"All"]}],两个键值对 分别存放标题 和 对应内容数组
+(NSArray *)getGroupedDictionaryArray:(NSArray *)array;
2.类似@[{@"A":@[@"abandon",@"啊哈哈",@"All"]}], 一个键值对,key存放标题,value存放对应内容数组
+(NSArray *)getGroupedDictionaryArray:(NSArray *)array indexKey:(NSString *)indexKey arrayKey:(NSString *)arrayKey;
4.由于本类中很多都是静态方法,内部无法使用非静态方法,并且很多是对NSString字符串的处理,
因为我在本类中写了一个分类,用于字符串的首字母类型判断,截取、转换等操作。
5.成果 :
NSArray *array = [YFGroupedData getGroupedDictionaryArray:[@"Once",@"Begin Again",@"hello wolrd",@"~",@"Windows",@"Lumia",@"苏菲",@"Yvan Wang",@"超能陆战队",@"Angry birds",@")",@"%",@"Windows Phone",@"950XL",@"速度与激情",@"1520",@"Titanic",@"霸王别姬",@"Captain America",@"World of Warcraft",@"星际穿越",@"反叛的鲁鲁修",@"消失的爱人",@"The Love of Siam",@"荷尔蒙",@"爱·回家",@"7号房的礼物",@"春夏秋冬又一春",@"The Little Prince",@"太阳的后裔",@"非首脑会谈",@"初恋这件小事",@"",@"Baby And Me",@"疯狂动物城",@"Old Boy",@"我的女孩",@"奔跑吧兄弟",@"我滴个神啊"]indexKey:kIndexKey arrayKey:kArrayKey];
来看看部分截图 为空的已经划分为✿部分了。
到此完成!
Github:YFGroupedData
以上是关于IOS将一组包含中文的数据按照#ABC...Z?分组的主要内容,如果未能解决你的问题,请参考以下文章