C # - самый простой способ удалить первое вхождение подстроки из другой строки - PullRequest
70 голосов
/ 04 февраля 2010

Мне нужно удалить первое (и ТОЛЬКО первое) вхождение строки из другой строки.

Вот пример замены строки "\\Iteration". Это:

ProjectName\\Iteration\\Release1\\Iteration1

станет таким:

ProjectName\\Release1\\Iteration1

Вот код, который делает это:

const string removeString = "\\Iteration";
int index = sourceString.IndexOf(removeString);
int length = removeString.Length;
String startOfString = sourceString.Substring(0, index);
String endOfString = sourceString.Substring(index + length);
String cleanPath = startOfString + endOfString;

Это похоже на большой код.

Итак, мой вопрос таков: есть ли более чистый / более читаемый / более краткий способ сделать это?

Ответы [ 7 ]

127 голосов
/ 04 февраля 2010
int index = sourceString.IndexOf(removeString);
string cleanPath = (index < 0)
    ? sourceString
    : sourceString.Remove(index, removeString.Length);
24 голосов
/ 04 февраля 2010
string myString = sourceString.Remove(sourceString.IndexOf(removeString),removeString.Length);

РЕДАКТИРОВАТЬ: @OregonGhost правильно. Я сам разбивал сценарий на условные выражения, чтобы проверить, не произошло ли это, но я действовал в предположении, что по каким-то требованиям строки должны принадлежать друг другу. Вполне возможно, что правила обработки исключений, которые требуются бизнесу, должны уловить эту возможность. Я сам использовал бы несколько дополнительных строк для выполнения условных проверок, а также для того, чтобы сделать его немного более читаемым для начинающих разработчиков, которые могут не уделить время для его тщательного прочтения.

15 голосов
/ 04 августа 2015
sourceString.Replace(removeString, "");
10 голосов
/ 04 февраля 2010

Написал быстрый тест TDD для этого

    [TestMethod]
    public void Test()
    {
        var input = @"ProjectName\Iteration\Release1\Iteration1";
        var pattern = @"\\Iteration";

        var rgx = new Regex(pattern);
        var result = rgx.Replace(input, "", 1);

        Assert.IsTrue(result.Equals(@"ProjectName\Release1\Iteration1"));
    }

rgx.Replace (вход, "", 1); говорит искать во входных данных что-либо, совпадающее с шаблоном, с "", 1 раз.

6 голосов
/ 04 февраля 2010

Вы можете использовать метод расширения для развлечения. Обычно я не рекомендую присоединять методы расширения к такому классу общего назначения, как string, но, как я уже сказал, это весело. Я позаимствовал ответ @ Люка, поскольку нет смысла заново изобретать колесо.

[Test]
public void Should_remove_first_occurrance_of_string() {

    var source = "ProjectName\\Iteration\\Release1\\Iteration1";

    Assert.That(
        source.RemoveFirst("\\Iteration"),
        Is.EqualTo("ProjectName\\Release1\\Iteration1"));
}

public static class StringExtensions {
    public static string RemoveFirst(this string source, string remove) {
        int index = source.IndexOf(remove);
        return (index < 0)
            ? source
            : source.Remove(index, remove.Length);
    }
}
2 голосов
/ 04 февраля 2010

Я определенно согласен, что это идеально подходит для метода расширения, но я думаю, что его можно немного улучшить.

public static string Remove(this string source, string remove,  int firstN)
    {
        if(firstN <= 0 || string.IsNullOrEmpty(source) || string.IsNullOrEmpty(remove))
        {
            return source;
        }
        int index = source.IndexOf(remove);
        return index < 0 ? source : source.Remove(index, remove.Length).Remove(remove, --firstN);
    }

Это немного рекурсии, что всегда весело.

Вот также простой юнит-тест:

   [TestMethod()]
    public void RemoveTwiceTest()
    {
        string source = "look up look up look it up";
        string remove = "look";
        int firstN = 2;
        string expected = " up  up look it up";
        string actual;
        actual = source.Remove(remove, firstN);
        Assert.AreEqual(expected, actual);

    }
1 голос
/ 03 октября 2018

Если вы хотите простой способ решить эту проблему. (Может использоваться как расширение)

См. Ниже:

    public static string RemoveFirstInstanceOfString(this string value, string removeString)
    {
        int index = value.IndexOf(removeString, StringComparison.Ordinal);
        return index < 0 ? value : value.Remove(index, removeString.Length);
    }

Использование:

    string valueWithPipes = "| 1 | 2 | 3";
    string valueWithoutFirstpipe = valueWithPipes.RemoveFirstInstanceOfString("|");
    //Output, valueWithoutFirstpipe = " 1 | 2 | 3";

Вдохновлен и изменен ответом @ LukeH's и @ Mike.

Не забудьте StringComparison.Ordinal, чтобы избежать проблем с настройками культуры https://www.jetbrains.com/help/resharper/2018.2/StringIndexOfIsCultureSpecific.1.html

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...