神奇的字符串翻译NSDataDetector

前言

我们经常需要对一个字符串内容的类型进行判断,是否是地址,电话或者URL.如果手动去处理,经常容易忽略一些判断规则的细节,导致结果不准确.

系统给我们提供了一个类NSDataDetector,它可以用于判断字符串内部内容的类型,而且,一段字符串中如果同时含有电话,地址,URL等,也可以区分出来,真是神奇的字符串翻译,下面我们就对这个magic class进行详细介绍.

NSDataDetector

NSDataDetectorNSRegularExpression的子类.看一下官方文档的介绍

A specialized regular expression object that matches natural language text for predefined data patterns

NSDataDetector是用于匹配自然语言字符串来确定对象类型的特殊的正则表达式对象.

NSDataDetector用法

目前NSDataDetector类可以用于匹配的类型有:日期, 地址, 链接, 手机号, 物流信息.匹配结果通过返回 NSTextCheckingResult 对象来表述.

NSTextCheckingResult

NSTextCheckingResult是表示匹配结果的类,需要注意的是NSTextCheckingResult可以用来表示NSRegularExpressionNSDataDetector的结果,但是通过NSDataDetector匹配返回的NSTextCheckingResult和父类NSRegularExpression返回的是不同的.

必要属性,存在于所有的解析结果中.

@property (readonly) NSTextCheckingType resultType;
@property (readonly) NSRange range;

可选属性,只有在匹配到特定类型时,某些属性才会有值.

@property (nullable, readonly, copy) NSOrthography *orthography;
@property (nullable, readonly, copy) NSArray<NSDictionary<NSString *, id> *> *grammarDetails;
@property (nullable, readonly, copy) NSDate *date;
@property (nullable, readonly, copy) NSTimeZone *timeZone;
@property (readonly) NSTimeInterval duration;
@property (nullable, readonly, copy) NSDictionary<NSTextCheckingKey, NSString *> *components API_AVAILABLE(macos(10.7), ios(4.0), watchos(2.0), tvos(9.0));
@property (nullable, readonly, copy) NSURL *URL;
@property (nullable, readonly, copy) NSString *replacementString;
@property (nullable, readonly, copy) NSArray<NSString *> *alternativeStrings API_AVAILABLE(macos(10.9), ios(7.0), watchos(2.0), tvos(9.0));
@property (nullable, readonly, copy) NSRegularExpression *regularExpression API_AVAILABLE(macos(10.7), ios(4.0), watchos(2.0), tvos(9.0));
@property (nullable, readonly, copy) NSString *phoneNumber API_AVAILABLE(macos(10.7), ios(4.0), watchos(2.0), tvos(9.0));

提到了匹配的类型,我们来看一下NSTextCheckingType resultType都包含哪些类型.

typedef NS_OPTIONS(uint64_t, NSTextCheckingType) {    // a single type
    NSTextCheckingTypeOrthography           = 1ULL << 0,            // language identification
    NSTextCheckingTypeSpelling              = 1ULL << 1,            // spell checking
    NSTextCheckingTypeGrammar               = 1ULL << 2,            // grammar checking
    NSTextCheckingTypeDate                  = 1ULL << 3,            // date/time detection
    NSTextCheckingTypeAddress               = 1ULL << 4,            // address detection
    NSTextCheckingTypeLink                  = 1ULL << 5,            // link detection
    NSTextCheckingTypeQuote                 = 1ULL << 6,            // smart quotes
    NSTextCheckingTypeDash                  = 1ULL << 7,            // smart dashes
    NSTextCheckingTypeReplacement           = 1ULL << 8,            // fixed replacements, such as copyright symbol for (c)
    NSTextCheckingTypeCorrection            = 1ULL << 9,            // autocorrection
    NSTextCheckingTypeRegularExpression API_AVAILABLE(macos(10.7), ios(4.0), watchos(2.0), tvos(9.0))  = 1ULL << 10,           // regular expression matches
    NSTextCheckingTypePhoneNumber API_AVAILABLE(macos(10.7), ios(4.0), watchos(2.0), tvos(9.0))        = 1ULL << 11,           // phone number detection
    NSTextCheckingTypeTransitInformation API_AVAILABLE(macos(10.7), ios(4.0), watchos(2.0), tvos(9.0)) = 1ULL << 12            // transit (e.g. flight) info detection
};

NSDataDetector能匹配的类型日期, 地址, 链接, 手机号, 物流信息.只有

NSTextCheckingTypeDate
NSTextCheckingTypeAddress
NSTextCheckingTypeLink
NSTextCheckingTypePhoneNumber
NSTextCheckingTypeTransitInformation

NSDataDetector返回的结果肯定属于dataDetectorTypes类型的某一种,并且根据结果类型的不同,匹配对应的属性.比如
日期类型NSTextCheckingTypeDate会包含NSDateNSTimeZone类型的timeZoneduration.
链接类型NSTextCheckingTypeLink会包含NSURL等.

示例

以下代码通过NSDataDetector来判断手机号URL,如果失败会返回error.

NSError *error = nil; 
NSDataDetector *detector = [NSDataDetector dataDetectorWithTypes: NSTextCheckingTypeLink | NSTextCheckingTypePhoneNumber error:&error];

NSDataDetector实例创建后,可以通过NSRegularExpression的方法numberOfMatches(in:options:range:)来获得字符串在某个rang内匹配的个数.

NSUInteger numberOfMatches = [detector numberOfMatchesInString:string options:0 range:NSMakeRange(0, [string length])];

如果只是想获取整个字符串的第一个匹配对象,numberOfMatches(in:options:range:)方法就可以.但是类型判断不同于正则表达式的地方在于,客户端会更注重分析后的附加信息.也就是NSTextCheckingResult对应的可选属性.

result的附加信息取决于其所属的类型.如果是NSTextCheckingTypeLink类型,那么属性URL就是关键的附加信息.如果是NSTextCheckingTypePhoneNumber类型,则phoneNumber是附加信息.

NSDateDetector的两个方法 matches(in:options:range:)firstMatch(in:options:range:)都可以用于获取解析结果.不同的是matches(in:options:range:)会返回所有的匹配结果,而firstMatch(in:options:range:)只会返回第一个匹配结果.

下面的代码段就是获取字符串中所有的链接电话号码的匹配结果.

NSArray *matches = [detector matchesInString:string options:0 range:NSMakeRange(0, [string length])]; 
for (NSTextCheckingResult *match in matches) {
     NSRange matchRange = [match range]; 
     if ([match resultType] == NSTextCheckingTypeLink) { 
        NSURL *url = [match URL]; 
     } else if ([match resultType] == NSTextCheckingTypePhoneNumber) { 
        NSString *phoneNumber = [match phoneNumber];
   } 
}

NSRegularExpression类的enumerator方法是匹配中最常用的方法.该方法运行我们多次去匹配一个字符串,在回调中自由地实现各种定制化需求,并且可以随时控制遍历停止的时机.通过block回调的属性BOOL *stop可以停止遍历.

下面是一个在匹配了特定次数后停止遍历的一个示例,可以用作参考.

__block NSUInteger count = 0;
 [detector enumerateMatchesInString:string 
                                options:0 
                                range:NSMakeRange(0, [string length])    
                                usingBlock:^(NSTextCheckingResult *match, NSMatchingFlags flags, BOOL *stop) { 
       NSRange matchRange = [match range];
       if ([match resultType] == NSTextCheckingTypeLink) { 
            NSURL *url = [match URL];
        } else if ([match resultType] == NSTextCheckingTypePhoneNumber) {
            NSString *phoneNumber = [match phoneNumber]; 
        }
         if (++count >= 100) *stop = YES;
 }];

注意

NSDataDetector只适用于解析特定类型的自然语言.只使用一定的情况匹配,并不是全能的.

如果文本具有特殊的格式,一个使用对应的系统提供的类型转换formatter来处理.比如想解析时间戳类型的数据,就应该使用DateFormatter类来解析,并输出NSDate对象.

如果文本是特定的类型如XMLJSON,那么应该先摘取自然语言,使用
XMLParserJSONSerialization进行解析.

相关文档

NSHipster的NSDataDetector

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容