获得两个NSStrings之间的差异

时间:2013-05-12 10:11:38

标签: objective-c string nsstring string-comparison

我有两个NSString,我希望得到它们之间的差异。

让我举个例子:

NSString *pippo = @"pippo";
NSString *pippone = @"pippone";

我想在两个实例中获取不同的字符。我怎么能这样做?

3 个答案:

答案 0 :(得分:2)

虽然可以使用更简单的算法处理几个特殊情况(例如后缀),但是处理一般情况,您可以:

1 /首先提取一个共同的序列。除非您的字符串很长,否则最常见的子序列算法将是您的选择(LCS,请参阅Here's)。

您可以在此处找到LCS的obj-c和swift实现:https://en.wikipedia.org/wiki/Longest_common_subsequence_problem

来自LCS,很容易得到差异,例如例如http://jakubturek.pl/blog/2015/06/27/using-swifts-string-type-with-care/。 请参阅下面的obj-C实现(NSString类,假设我们有最长的CommonSubsequence:来自上面的链接):

- (NSArray *) lcsDiff:(NSString *)string
{
    NSString *lcs = [self longestCommonSubsequence:string];
    NSUInteger l1 = [self length];
    NSUInteger l2 = [string length];
    NSUInteger lc = [lcs length];
    NSUInteger idx1 = 0;
    NSUInteger idx2 = 0;
    NSUInteger idxc = 0;
    NSMutableString *s1 = [[NSMutableString alloc]initWithCapacity:l1];
    NSMutableString *s2 = [[NSMutableString alloc]initWithCapacity:l2];
    NSMutableArray *res = [NSMutableArray arrayWithCapacity:10];
    for (;;) {
        if (idxc >= lc) break;
        unichar c1 = [self characterAtIndex:idx1];
        unichar c2 = [string characterAtIndex:idx2];
        unichar cc = [lcs characterAtIndex:idxc];
        if ((c1==cc) && (c2 == cc)) {
            if ([s1 length] || [s2 length]) {
                NSArray *e = @[ s1, s2];
                [res addObject:e];
                s1 = [[NSMutableString alloc]initWithCapacity:l1];
                s2 = [[NSMutableString alloc]initWithCapacity:l1];
            }
            idx1++; idx2++; idxc++;
            continue;
        }
        if (c1 != cc) {
            [s1 appendString:[NSString stringWithCharacters:&c1 length:1]];
            idx1++;
        }
        if (c2 != cc) {
            [s2 appendString:[NSString stringWithCharacters:&c2 length:1]];
            idx2++;
        }
    }
    if (idx1<l1) {
        [s1 appendString:[self substringFromIndex:idx1]];
    }
    if (idx2<l2) {
        [s2 appendString:[string substringFromIndex:idx2]];
    }
    if ([s1 length] || [s2 length]) {
        NSArray *e = @[ s1, s2];
        [res addObject:e];
    }
    return res;
}

代码将返回一个数组,其中包含一个元素(string1,string2),用于两个字符串之间的每个差异,例如: [@“abcXefY”lcsDiff:@“aZbcKef”]将返回@[ @[ @"", @"Z"], @[ @"X", @"K"], @[ @"Y", @""]];

答案 1 :(得分:0)

你可以使用一些regEx:

我做了一次这个函数,用你想要的字符串/模式替换另一个字符串中的字符串。

-(NSString *)replaceInString:(NSString *)chaine :(NSString *)pattern
             :(NSString *)template
{
NSMutableString *chaineMutable = [[NSMutableString alloc] initWithString:chaine];
NSRegularExpression *regex = [[NSRegularExpression alloc] init];

regex = [NSRegularExpression regularExpressionWithPattern:pattern
             options:NSRegularExpressionCaseInsensitive error:NULL];

[regex replaceMatchesInString:(NSMutableString *)chaineMutable
             options:NSMatchingReportProgress range:NSMakeRange(0, [chaine length])
             withTemplate:template];

NSString *returnedString = [[NSString alloc] initWithString:chaineMutable];

return returnedString;
}

然后你可以调用它:

NSString *difference = [self replaceInString:pippone :pippo :@""];

所以你将“pippone”中的“pippo”替换为“”(无),所以结果将是“ne”。

希望这有帮助。

答案 2 :(得分:-2)

更新的答案:

-(void)getDifferenceBetweenString:(NSString *)firstString secondString:(NSString *)secondString{
    NSString *longerString = @"";
    NSString *shorterString = @"";
    if (firstString.length >= secondString.length) {
        longerString = firstString;
        shorterString = secondString;
    }else{
        longerString = secondString;
        shorterString = firstString;

    }
    NSArray *shorterArray = [self convertToArray:shorterString];
    NSArray *longerArray = [self convertToArray:longerString];
    NSMutableArray *differenceLettersArray = [NSMutableArray new];
    for (NSString * letter in shorterArray) {
        if (![longerString containsString:letter]) {
            if (![differenceLettersArray containsObject:letter]) {
                [differenceLettersArray addObject:letter];

            }
        }
    }


    for (NSString * letter in longerArray) {
        if (![shorterString containsString:letter]) {
            if (![differenceLettersArray containsObject:letter]) {
                [differenceLettersArray addObject:letter];

            }
        }
    }
    NSLog(@"differences = %@",differenceLettersArray);
}

- (NSArray *)convertToArray:(NSString *)string {
    NSMutableArray *arr = [[NSMutableArray alloc] init];
    NSUInteger i = 0;
    while (i < string.length) {
        NSRange range = [string rangeOfComposedCharacterSequenceAtIndex:i];
        NSString *chStr = [string substringWithRange:range];
        [arr addObject:chStr];
        i += range.length;
    }

    return arr;
}

使用如下:

[self getDifferenceBetweenString:@"pippo" secondString:@"pippone"];