метод сравнения строк c #, возвращающий индекс первого несоответствия - PullRequest
18 голосов
/ 14 декабря 2011

Существует ли метод сравнения строк, который будет возвращать значение, основанное на первом вхождении несовпадающего символа между двумя строками?

т.е.

string A = "1234567890"

string B = "1234567880"

Я хотел бы получитьобратное значение, которое позволило бы мне увидеть, что первое вхождение совпадающего разрыва - A [8]

Ответы [ 6 ]

7 голосов
/ 14 декабря 2011
/// <summary>
/// Gets a first different char occurence index
/// </summary>
/// <param name="a">First string</param>
/// <param name="b">Second string</param>
/// <param name="handleLengthDifference">
/// If true will return index of first occurence even strings are of different length
/// and same-length parts are equals otherwise -1
/// </param>
/// <returns>
/// Returns first difference index or -1 if no difference is found
/// </returns>
public int GetFirstBreakIndex(string a, string b, bool handleLengthDifference)
{
    int equalsReturnCode = -1;
    if (String.IsNullOrEmpty(a) || String.IsNullOrEmpty(b))
    {
        return handleLengthDifference ? 0 : equalsReturnCode;
    }

    string longest = b.Length > a.Length ? b : a;
    string shorten = b.Length > a.Length ? a : b;    
    for (int i = 0; i < shorten.Length; i++)
    {
        if (shorten[i] != longest[i])
        {
            return i;
        }
    }

    // Handles cases when length is different (a="1234", b="123")
    // index=3 would be returned for this case
    // If you do not need such behaviour - just remove this
    if (handleLengthDifference && a.Length != b.Length)
    {
        return shorten.Length;
    }

    return equalsReturnCode;
}
3 голосов
/ 14 декабря 2011

Метод расширения, аналогичный приведенному ниже, сделает эту работу:

public static int Your_Name_Here(this string s, string other) 
{
    string first = s.Length < other.Length ? s : other;
    string second = s.Length > other.Length ? s : other;

    for (int counter = 0; counter < first.Length; counter++)
    {
        if (first[counter] != second[counter])
        {
            return counter;
        }
    }
    return -1;
}
2 голосов
/ 14 декабря 2011

Если у вас установлен .net 4.0, это может быть:

    string A = "1234567890";
    string B = "1234567880";

    char? firstocurrence = A.Zip(B, (p, q) => new { A = p, B = q })
        .Where(p => p.A != p.B)
        .Select(p => p.A)
        .FirstOrDefault();

edit:

Хотя, если вам нужна позиция:

    int? firstocurrence = A.Zip(B, (p, q) => new { A = p, B = q })
            .Where(p => p.A != p.B)
            .Select((p, i) => i)
            .FirstOrDefault();
2 голосов
/ 14 декабря 2011

Не то, чтобы я знал, но это довольно тривиально:

public static int FirstUnmatchedIndex(this string x, string y)
{
  if(x == null || y == null)
    throw new ArgumentNullException();
  int count = x.Length;
  if(count > y.Length)
    return FirstUnmatchedIndex(y, x);
  if(ReferenceEquals(x, y))
    return -1;
  for(idx = 0; idx != count; ++idx)
    if(x[idx] != y[idx])
      return idx;
  return count == y.Length? -1 : count;
}

Это простое порядковое сравнение. Порядковое сравнение без учета регистра - это легкое изменение, но основа культуры сложно определить; «Weißbier» не соответствует «WEISSBIERS» в конце S во второй строке, но считается ли это позицией 8 или 9?

0 голосов
/ 04 ноября 2015

Эта тема является дубликатом Сравнение строк и первое место, где они отличаются от друг друга , который содержит лучшее решение в одну строку с использованием Linq

0 голосов
/ 14 декабря 2011

Можно написать расширение строки, например

public static class MyExtensions
{
    public static IList<char> Mismatch(this string str1, string str2)
    {
        var char1 = str1.ToCharArray();
        var char2 = str2.ToCharArray();
        IList<Char> Resultchar= new List<char>();
        for (int i = 0; i < char2.Length;i++ )
        {
            if (i >= char1.Length || char1[i] != char2[i])
                Resultchar.Add(char2[i]);
        }
        return Resultchar;
    }
}

Использовать его как

var r = "1234567890".Mismatch("1234567880");

Это не оптимизированный алгоритм для обнаружения несоответствия.

Если вызаинтересованы только найти первое несоответствие,

public static Char FirstMismatch(this string str1, string str2)
        {
            var char1 = str1.ToCharArray();
            var char2 = str2.ToCharArray();             
            for (int i = 0; i < char2.Length;i++ )
            {
                if (i >= char1.Length || char1[i] != char2[i])
                    return char2[i];
            }
            return ''c;
        }
...