仅使用Regex替换某些组

Osc*_*ros 179 c# regex replace

我们假设我有以下正则表达式:

-(\d+)-
Run Code Online (Sandbox Code Playgroud)

我想更换,使用C#,1组(\d+)AA,获得:

-AA-
Run Code Online (Sandbox Code Playgroud)

现在我用以下方法替换它:

var text = "example-123-example";
var pattern = @"-(\d+)-";
var replaced = Regex.Replace(text, pattern, "-AA-"); 
Run Code Online (Sandbox Code Playgroud)

但我真的不喜欢这个,因为如果我改变模式来匹配_(\d+)_,我也必须改变替换字符串_AA_,这是违反DRY原则的.

我正在寻找类似的东西:

保持匹配的文本的确切方式,但更改组1 this text和组2 another text...

编辑:
这只是一个例子.我只是在寻找一种做我上面所说的通用方法.

它应该适用于:

anything(\d+)more_text 以及你能想象到的任何模式.

我想要做的只是替换组,并保持匹配的其余部分.

blu*_*ume 287

一个好主意可能是将所有内容封装在组内,无论是否需要识别它们.这样你就可以在替换字符串中使用它们.例如:

var pattern = @"(-)(\d+)(-)";
var replaced = Regex.Replace(text, pattern, "$1AA$3"); 
Run Code Online (Sandbox Code Playgroud)

或使用MatchEvaluator:

var replaced = Regex.Replace(text, pattern, m => m.Groups[1].Value + "AA" + m.Groups[3].Value);
Run Code Online (Sandbox Code Playgroud)

另一种方式,略显凌乱,可能是使用lookbehind/lookahead:

(?<=-)(\d+)(?=-)

  • 我编辑了你的答案以提供更多信息,但你所说的完全正确.**不知道我怎么错过了我可以将所有内容都放在组内,无论是否使用它们**`:).在我看来,这个解决方案比使用前瞻和外观更好更清洁. (16认同)
  • 另外值得注意的是 - 如果您的替换文本以数字开头,则第一个解决方案("$ 1AA $ 3")将无法按预期工作! (9认同)
  • @OscarMederos,您也可以使用非捕获组-适用于不使用的组。在`(?:foo)(bar)`中,`$ 1`将替换`bar`。[更多详细信息](/sf/ask/245873001/) (2认同)

Luk*_*keH 32

你可以使用lookahead和lookbehind来做到这一点:

var pattern = @"(?<=-)\d+(?=-)";
var replaced = Regex.Replace(text, pattern, "AA"); 
Run Code Online (Sandbox Code Playgroud)


Dan*_*rth 17

我也需要这个,我为它创建了以下扩展方法:

public static class RegexExtensions
{
    public static string ReplaceGroup(
        this Regex regex, string input, string groupName, string replacement)
    {
        return regex.Replace(
            input,
            m =>
            {
                var group = m.Groups[groupName];
                var sb = new StringBuilder();
                var previousCaptureEnd = 0;
                foreach (var capture in group.Captures.Cast<Capture>())
                {
                    var currentCaptureEnd =
                        capture.Index + capture.Length - m.Index;
                    var currentCaptureLength =
                        capture.Index - m.Index - previousCaptureEnd;
                    sb.Append(
                        m.Value.Substring(
                            previousCaptureEnd, currentCaptureLength));
                    sb.Append(replacement);
                    previousCaptureEnd = currentCaptureEnd;
                }
                sb.Append(m.Value.Substring(previousCaptureEnd));

                return sb.ToString();
            });
    }
}
Run Code Online (Sandbox Code Playgroud)

用法:

var input = @"[assembly: AssemblyFileVersion(""2.0.3.0"")][assembly: AssemblyFileVersion(""2.0.3.0"")]";
var regex = new Regex(@"AssemblyFileVersion\(""(?<version>(\d+\.?){4})""\)");


var result = regex.ReplaceGroup(input , "version", "1.2.3");
Run Code Online (Sandbox Code Playgroud)

结果:

[assembly: AssemblyFileVersion("1.2.3")][assembly: AssemblyFileVersion("1.2.3")]
Run Code Online (Sandbox Code Playgroud)


小智 13

如果您不想更改模式,可以使用匹配组的"组索引"和"长度"属性.

var text = "example-123-example";
var pattern = @"-(\d+)-";
var regex = new RegEx(pattern);
var match = regex.Match(text);

var firstPart = text.Substring(0,match.Groups[1].Index);    
var secondPart = text.Substring(match.Groups[1].Index + match.Groups[1].Length);
var fullReplace = firstPart + "AA" + secondPart;
Run Code Online (Sandbox Code Playgroud)


cur*_*ius 6

这是另一个不错的清洁选项,不需要更改您的模式。

        var text = "example-123-example";
        var pattern = @"-(\d+)-";

        var replaced = Regex.Replace(text, pattern, (_match) =>
        {
            Group group = _match.Groups[1];
            string replace = "AA";
            return String.Format("{0}{1}{2}", _match.Value.Substring(0, group.Index - _match.Index), replace, _match.Value.Substring(group.Index - _match.Index + group.Length));
        });
Run Code Online (Sandbox Code Playgroud)