С# String заменить словарем
У меня есть строка, на которой мне нужно сделать некоторые замены. У меня есть Dictionary<string, string>
, где я определяю пары поиска и замены. Я создал следующие методы расширения для выполнения этой операции:
public static string Replace(this string str, Dictionary<string, string> dict)
{
StringBuilder sb = new StringBuilder(str);
return sb.Replace(dict).ToString();
}
public static StringBuild Replace(this StringBuilder sb,
Dictionary<string, string> dict)
{
foreach (KeyValuePair<string, string> replacement in dict)
{
sb.Replace(replacement.Key, replacement.Value);
}
return sb;
}
Есть ли лучший способ сделать это?
Ответы
Ответ 1
Если данные являются токенированными (т.е. "Дорогое имя $$, то из $date $ваш баланс равен $amount $" ), тогда может быть полезно Regex
:
static readonly Regex re = new Regex(@"\$(\w+)\$", RegexOptions.Compiled);
static void Main() {
string input = @"Dear $name$, as of $date$ your balance is $amount$";
var args = new Dictionary<string, string>(
StringComparer.OrdinalIgnoreCase) {
{"name", "Mr Smith"},
{"date", "05 Aug 2009"},
{"amount", "GBP200"}
};
string output = re.Replace(input, match => args[match.Groups[1].Value]);
}
Однако, без чего-то подобного, я ожидаю, что ваш цикл Replace
, вероятно, будет примерно таким же, как вы можете, без крайностей. Если он не обозначен, возможно, его профиль; является Replace
на самом деле проблемой?
Ответ 2
Сделайте это с помощью Linq:
var newstr = dict.Aggregate(str, (current, value) =>
current.Replace(value.Key, value.Value));
dict - это ваш поиск-замените пары, определенные в словаре.
str - это ваша строка, в которой вам нужно выполнить некоторые замены.
Ответ 3
Кажется разумным для меня, кроме одного: он чувствителен к порядку. Например, возьмите строку ввода "$ x $y" и заменяющий словарь:
"$x" => "$y"
"$y" => "foo"
Результаты замены являются либо "foo foo", либо "$ y foo", в зависимости от того, какая замена выполняется в первую очередь.
Вместо этого вы можете управлять порядком, используя List<KeyValuePair<string, string>>
. Альтернативой является прогулка по строке, чтобы вы не потребляли замены при последующих операциях замены. Скорее всего, это будет намного сложнее.
Ответ 4
Здесь приведена небольшая рефакторированная версия замечательного ответа @Marc, чтобы сделать функциональность доступной как метод расширения для Regex:
static void Main()
{
string input = @"Dear $name$, as of $date$ your balance is $amount$";
var args = new Dictionary<string, string>(StringComparer.OrdinalIgnoreCase);
args.Add("name", "Mr Smith");
args.Add("date", "05 Aug 2009");
args.Add("amount", "GBP200");
Regex re = new Regex(@"\$(\w+)\$", RegexOptions.Compiled);
string output = re.replaceTokens(input, args);
// spot the LinqPad user // output.Dump();
}
public static class ReplaceTokensUsingDictionary
{
public static string replaceTokens(this Regex re, string input, IDictionary<string, string> args)
{
return re.Replace(input, match => args[match.Groups[1].Value]);
}
}
Ответ 5
при использовании решения Marc Gravell RegEx сначала проверьте, доступен ли токен с использованием i.e. ContainsKey, чтобы предотвратить ошибки KeyNotFoundException:
string output = re.Replace(zpl, match => { return args.ContainsKey(match.Groups[1].Value) ? arg[match.Groups[1].Value] : match.Value; });
при использовании следующего слегка модифицированного образца кода (первый параметр имеет другое имя):
var args = new Dictionary<string, string>(
StringComparer.OrdinalIgnoreCase)
{
{"nameWRONG", "Mr Smith"},
{"date", "05 Aug 2009"},
{"AMOUNT", "GBP200"}
};
это приводит к следующему:
"Дорогое $name $, по состоянию на 05 августа 2009 года ваш баланс равен GBP200"
Ответ 6
Здесь вы находитесь:
public static class StringExm
{
public static String ReplaceAll(this String str, KeyValuePair<String, String>[] map)
{
if (String.IsNullOrEmpty(str))
return str;
StringBuilder result = new StringBuilder(str.Length);
StringBuilder word = new StringBuilder(str.Length);
Int32[] indices = new Int32[map.Length];
for (Int32 characterIndex = 0; characterIndex < str.Length; characterIndex++)
{
Char c = str[characterIndex];
word.Append(c);
for (var i = 0; i < map.Length; i++)
{
String old = map[i].Key;
if (word.Length - 1 != indices[i])
continue;
if (old.Length == word.Length && old[word.Length - 1] == c)
{
indices[i] = -old.Length;
continue;
}
if (old.Length > word.Length && old[word.Length - 1] == c)
{
indices[i]++;
continue;
}
indices[i] = 0;
}
Int32 length = 0, index = -1;
Boolean exists = false;
for (int i = 0; i < indices.Length; i++)
{
if (indices[i] > 0)
{
exists = true;
break;
}
if (-indices[i] > length)
{
length = -indices[i];
index = i;
}
}
if (exists)
continue;
if (index >= 0)
{
String value = map[index].Value;
word.Remove(0, length);
result.Append(value);
if (word.Length > 0)
{
characterIndex -= word.Length;
word.Length = 0;
}
}
result.Append(word);
word.Length = 0;
for (int i = 0; i < indices.Length; i++)
indices[i] = 0;
}
if (word.Length > 0)
result.Append(word);
return result.ToString();
}
}