4

我想要一个 NSRegularExpression– stringByReplacingMatchesInString:options:range:withTemplate:方法的变体,它采用块而不是模板。块的返回值将用作替换值。正如您可以想象的那样,这比模板更灵活。有点像/e在 Perl 正则表达式中使用修饰符。

所以我写了一个类别来添加方法。这就是我想出的:

@implementation NSRegularExpression (Block)

- (NSString *)stringByReplacingMatchesInString:(NSString *)string
                                       options:(NSMatchingOptions)options
                                         range:(NSRange)range
                                    usingBlock:(NSString* (^)(NSTextCheckingResult *result))block
{
    NSMutableString *ret = [NSMutableString string];
    NSUInteger pos = 0;

    for (NSTextCheckingResult *res in [self matchesInString:string options:options range:range]) {
        if (res.range.location > pos) {
            [ret appendString:[string substringWithRange:NSMakeRange(pos, res.range.location - pos)]];
        }
        pos = res.range.location + res.range.length;
        [ret appendString:block(res)];
    }
    if (string.length > pos) {
        [ret appendString:[string substringFromIndex:pos]];
    }
    return ret;
}

@end

这是我第一次尝试在 Objective C 中玩积木。感觉有点奇怪,但似乎效果很好。不过,我有几个问题:

  1. 这似乎是实现这种方法的明智方法吗?
  2. 有没有办法使用它来实现它的内部-enumerateMatchesInString:options:range:usingBlock: ?我试过了,但无法pos从块内分配。但是如果有办法让它工作,那么传递 NSMatchingFlags 和 BOOL 并以与该方法相同的方式处理它们会很酷。可以吗?

更新

感谢 Dave DeLong 的回答,我有了一个使用块的新版本:

@implementation NSRegularExpression (Block)

- (NSString *)stringByReplacingMatchesInString:(NSString *)string
                                       options:(NSMatchingOptions)options
                                         range:(NSRange)range
                                    usingBlock:(NSString * (^)(NSTextCheckingResult *result, NSMatchingFlags flags, BOOL *stop))block
{
    NSMutableString *ret = [NSMutableString string];
    __block NSUInteger pos = 0;

    [self enumerateMatchesInString:string options:options range:range usingBlock:^(NSTextCheckingResult *match, NSMatchingFlags flags, BOOL *stop)
    {
        if (match.range.location > pos) {
            [ret appendString:[string substringWithRange:NSMakeRange(pos, match.range.location - pos)]];
        }
        pos = match.range.location + match.range.length;
        [ret appendString:block(match, flags, stop)];
    }];
    if (string.length > pos) {
        [ret appendString:[string substringFromIndex:pos]];
    }
    return [NSString stringWithString:ret];
}

@end

效果很好,谢谢!

4

1 回答 1

6

能够pos从块内分配到就像更改声明一样简单:

NSUInteger pos = 0;

到:

__block NSUInteger pos = 0;

有关__block关键字的更多信息: __block变量

于 2010-12-18T00:07:42.957 回答