NSCharacterSet 简单用法
NSCharacterSet 简单用法
NSMutableCharacterSet *base = [NSMutableCharacterSet lowercaseLetterCharacterSet]; //字母
NSCharacterSet *decimalDigit = [NSCharacterSet decimalDigitCharacterSet]; //十进制数字
[base formUnionWithCharacterSet:decimalDigit]; //字母加十进制
NSString *string = @"ax@d5s#@sfn$5`SF$$%x^(#e{]e";
//用上面的base隔开string然后组成一个数组,然后通过componentsJoinedByString,来连接成一个字符串
NSLog(@"%@",[[string componentsSeparatedByCharactersInSet:base] componentsJoinedByString:@"-"]);
[base invert]; //非 字母加十进制
NSLog(@"%@",[[string componentsSeparatedByCharactersInSet:base] componentsJoinedByString:@"-"]);
答应结果:
ax@d-s#@sfn$-`SF$$%x^(#e{]e
---------------------------------------------
正如之前提前过的,基础类库(Foundation)拥有最好的、功能也最全的string类的实现。
但是仅当程序员熟练掌握它时,一个string的实现才是真的好。所以本周,我们将浏览一些基础类库的string生态系统中经常用到且用错的重要组成部分:NSCharacterSet
。
如果你对什么是字符编码搞不清楚的话(即使你有很好的专业知识),那么你应该抓住这次机会反复阅读Joel Spolsky的这篇经典的文章"The Absolute Minimum Every Software Developer Absolutely, Positively Must Know About Unicode and Character Sets (No Excuses!)"。在头脑中保持新鲜感将对你理解我们将要探讨的话题非常有帮助。
NSCharacterSet
,以及它的可变版本NSMutableCharacterSet
,用面向对象的方式来表示一组Unicode字符。它经常与NSString
及NSScanner
组合起来使用,在不同的字符上做过滤、删除或者分割操作。为了给你提供这些字符是哪些字符的直观印象,请看看NSCharacterSet
提供的类方法:
alphanumericCharacterSet
capitalizedLetterCharacterSet
controlCharacterSet
decimalDigitCharacterSet
decomposableCharacterSet
illegalCharacterSet
letterCharacterSet
lowercaseLetterCharacterSet
newlineCharacterSet
nonBaseCharacterSet
punctuationCharacterSet
symbolCharacterSet
uppercaseLetterCharacterSet
whitespaceAndNewlineCharacterSet
whitespaceCharacterSet
与它的名字所表述的相反,NSCharacterSet
跟 NSSet
一点关系都没有。
虽然底层实现不太一样,但是 NSCharacterSet
在概念上跟 NSIndexSet
还有点相似的。NSIndexSet
,之前提到过,表示一个有序的不重复的无符号整数的集合。Unicode字符跟无符号整数类似,大致对应一些拼写表示。所以,一个 NSCharacterSet +lowercaseCharacterSet
字符集与一个包含97到122范围的 NSIndexSet
是等价的。
现在我们对理解 NSCharacterSet
的基本概念已经有了少许自信,让我们来看一些它的模式与反模式吧:
去掉空格
NSString -stringByTrimmingCharactersInSet:
是个你需要牢牢记住的方法。它经常会传入NSCharacterSet +whitespaceCharacterSet
或 +whitespaceAndNewlineCharacterSet
来删除输入字符串的头尾的空白符号。
需要重点注意的是,这个方法 仅仅 去除了 开头 和 结尾 的指定字符集中连续字符。这就是说,如果你想去除单词之间的额外空格,请看下一步。
挤压空格
假设你去掉字符串两端的多余空格之后,还想去除单词之间的多余空格,这里有个非常简便的方法:
NSString *string = @"Lorem ipsum dolar sit amet."; string = [string stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]]; NSArray *components = [string componentsSeparatedByCharactersInSet:[NSCharacterSet whitespaceCharacterSet]]; components = [components filteredArrayUsingPredicate:[NSPredicate predicateWithFormat:@"self <> ''"]]; string = [components componentsJoinedByString:@" "];
首先,删除字符串首尾的空格;然后用 NSString -componentsSeparatedByCharactersInSet:
在空格处将字符串分割成一个 NSArray
;再用一个 NSPredicate
去除空串;最后,用 NSArray -componentsJoinedByString:
用单个空格符将数组重新拼成字符串。注意:这种方法仅适用于英语这种用空格分割的语言。
现在看看反模式吧。请先看看 the answers to this question on StackOverflow。
在写这篇文章的时候,排行第二的正确答案有 58 个顶和 2 个踩。排行第一的有 84 个顶和 24 个踩。
如今,排名第一的答案却不是正确答案是不太正常的,但是这个问题已经破了不重复答案数(10个)的记录,同时也破了不重复、完全错误的答案数(9个)的记录。
言归正传,这里有 9 个错误答案:
- "Use
stringByTrimmingCharactersInSet
" - 正如你所知道的,它只去掉首尾的空格。 - "Replace ' ' with ''" - 这个去除了所有的空格,劳而无功。
- "Use a regular expression" - 有点用,但它没有处理首尾的空格。用正则表达式有点大材小用了。
- "Use Regexp Lite" - 说真的,正则表达式真心没必要。同时为了这点功能增加第三方库很不值。
- "Use OgreKit" - 同上,添加了第三方库。
- "Split the string into components, iterate over them to find components with non-zero length, and then re-combine" - 很接近了,但是
componentsSeparatedByCharactersInSet:
已经让遍历变得没必要。 - "Replace two-space strings with single-space strings in a while loop" - 错误且浪费计算资源。
- "Manually iterate over each
unichar
in the string and useNSCharacterSet -characterIsMember:
" - 用了一个复杂到让人吃惊的程度的方法,却忘了标准库中已经有方法可以用。 - "Find and remove all of the tabs" - 有谁提到了制表符了?不过还是谢谢了吧。
我个人并不是想责怪回答问题的人——只是指出完成这个功能有多少种不同的方法,而这些方法有多少是完全错误的。
字符串分词
不要用 NSCharacterSet
来分词。 用 CFStringTokenizer
来替代它。
你用 componentsSeparatedByCharactersInSet:
来清理用户输入是可以谅解的,但是用它来做更复杂的事情,你将陷入痛苦的深渊。
为什么?请记住,语言并不是都用空格作为词的分界。虽然实际上以空格分界的语言使用非常广泛。但哪怕只算上中国和日本就已经有十多亿人,占了世界人口总量的 16%。
……即使是用空格分隔的语言,分词也有一些模棱两可的边界条件,特别是复合词汇和标点符号。
以上只为说明:如果你想将字符串分成有意义的单词,那么请用 CFStringTokenizer
(或者enumerateSubstringsInRange:options:usingBlock:
)吧。
从字符串解析数据
NSScanner
是个用以解析任意或半结构化的字符串的数据的类。当你为一个字符串创建一个扫描器时,你可以指定忽略哪些字符,这样可以避免那些字符以各种各样的方式被包含到解析出来的结果中。
例如,你想从这样一个字符串中解析出开门时间:
Mon-Thurs: 8:00 - 18:00
Fri: 7:00 - 17:00
Sat-Sun: 10:00 - 15:00
你会 enumerateLinesUsingBlock:
并像这样用一个 NSScanner
来解析:
let skippedCharacters = NSMutableCharacterSet() skippedCharacters.formIntersectionWithCharacterSet(NSCharacterSet.punctuationCharacterSet()) skippedCharacters.formIntersectionWithCharacterSet(NSCharacterSet.whitespaceCharacterSet()) string.enumerateLines { (line, _) in let scanner = NSScanner(string: line) scanner.charactersToBeSkipped = skippedCharacters var startDay, endDay: NSString? var startHour: Int = 0 var startMinute: Int = 0 var endHour: Int = 0 var endMinute: Int = 0 scanner.scanCharactersFromSet(NSCharacterSet.letterCharacterSet(), intoString: &startDay) scanner.scanCharactersFromSet(NSCharacterSet.letterCharacterSet(), intoString: &endDay) scanner.scanInteger(&startHour) scanner.scanInteger(&startMinute) scanner.scanInteger(&endHour) scanner.scanInteger(&endMinute) }
我们首先从空格字符集和标点符号字符集的并集构造了一个 NSMutableCharacterSet
。告诉 NSScanner
忽略这些字符以极大地减少解析这些字符的必要逻辑。
scanCharactersFromSet:
传入字母字符集得到每项中一星期内的开始和结束(可选)的天数。scanInteger
类似地,得到下一个连续的整型值。
NSCharacterSet
和 NSScanner
让你可以快速而充满自信地编码。这两者真是完美组合。
NSCharacterSet
是基础类库中字符串处理系统中的一员,可能是最容易被用错或是误解的一员。在脑中记住这些模式与反模式,你将不仅能做一些很有用的诸如管理空格及从字符串中读信息之类的事情,更重要的是,你将避免误入歧途。
如果“不出错”对一个 NSHipster 来说不是最重要的事情,那我也不想成为正确的了!
Ed. Speaking of (not) being wrong, the original version of this article contained errors in both code samples. These have since been corrected.
NSCharacterSet 简单用法相关推荐
- html.renderaction 控制器,Html.RenderAction简单用法
CATransition(os开发之画面切换) 的简单用法 CATransition 的简单用法 //引进CATransition 时要添加包"QuartzCore.framework&qu ...
- 反编译工具jad简单用法
反编译工具jad简单用法 下载地址: [url]http://58.251.57.206/down1?cid=B99584EFA6154A13E5C0B273C3876BD4CC8CE672& ...
- QCustomPlot的简单用法总结
QCustomPlot的简单用法总结 第一部分:QCustomPlot的下载与安装 第二部分:QCustomPlot在VS2013+QT下的使用 QCustomPlot的简单用法总结 写在前面, ...
- python matplotlib 简单用法
python matplotlib 简单用法 具体内容请参考官网 代码 import matplotlib.pyplot as plt import numpy as np # 支持中文 plt.rc ...
- Windump网络命令的简单用法
Windump网络命令的简单用法 大家都知道,unix系统下有个tcpdump的抓包工具,非常好用,是做troubleshooting的好帮手.其实在windows下也有一个类似的工作,叫windum ...
- Android TabLayout(选项卡布局)简单用法实例分析
本文实例讲述了Android TabLayout(选项卡布局)简单用法.分享给大家供大家参考,具体如下: 我们在应用viewpager的时候,经常会使用TabPageIndicator来与其配合.达到 ...
- shell expect的简单用法
为什么需要expect? 我们通过Shell可以实现简单的控制流功能,如:循环.判断等.但是对于需要交互的场合则必须通过人工来干预,有时候我们可能会需要实现和交互程序如 telnet服务器等进 ...
- Shellz中awk的简单用法
其实shell脚本的功能常常被低估.在实际应用中awk sed 等用法可以为shell提供更为强大的功能.下面我们将一下awk调用的简单方法进行了总结.方便同学们学习: awk的简单用法: 第一种调用 ...
- python装饰器实例-Python装饰器原理与简单用法实例分析
本文实例讲述了Python装饰器原理与简单用法.分享给大家供大家参考,具体如下: 今天整理装饰器,内嵌的装饰器.让装饰器带参数等多种形式,非常复杂,让人头疼不已.但是突然间发现了装饰器的奥秘,原来如此 ...
最新文章
- python简介怎么写-python怎么写函数
- boost::multiprecision模块debug_adaptor相关的测试程序
- LEANSOFT领航员 – 基于Docker的DevOps实战培训
- 如何使用mock应对测试所需随机数据
- proc_fs文件的操作
- OpenCV图像处理篇之边缘检測算子
- JAVA变量直接可以相加不,Java学习第一天
- mysql sql slave skip_MySQL的一次错误处理 SQL_SLAVE_SKIP_COUNTER
- 将背景为白色图片转为png格式的透明图片
- 全国近三成高考状元被曝有加分(图)
- HW红队攻防基础建设—C2 IP隐匿技术
- 专家通过六点考证唐村《李氏族谱》:辨别家谱真伪,有这些窍门
- 通过Python来调用Chrom浏览器,进入设定网页
- (转)一共81个,开源大数据处理工具汇总
- java饼图_java 百分比饼图的实现代码
- 牛客_求将一个数组分割为两个差值最小的部分
- Map根据Key值进行排序(升序加降序)
- 关于a21A11+a22A12+a23A13=0的证明
- html5版微博能收到私信吗,微博私信功能说明
- 腾讯云年终选购云服务器攻略!
热门文章
- Nginx中break和last的区别
- 善用assert函数
- 【资料下载】Python 第九讲——灵活运用docker,实现深度学习的环境搭建...
- Dapper and Repository Pattern in MVC
- iphone:解析html的第三库hpple初试
- Atom 备份神器 —— Sync Settings
- weblogic .NoClassDefFoundError: Could not initialize class sun.awt.X11Graphi
- 9-16 原生命令和redis-trib.rb对比
- 102.如何减轻缓存失效时上游服务的压力
- instanceof java 原理_JAVA中 instanceof 和 getClass() 区别小结