用正则表达式仅替换某些组


190

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

-(\d+)-

我想使用C#将Group 1 (\d+)替换为AA,以获得:

-AA-

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

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

但是我不是很喜欢这样,因为如果我改为更改模式以匹配_(\d+)_,那么我也将不得不替换替换字符串_AA_,这违反了DRY原则。

我正在寻找类似的东西:

使匹配的文本保持原样,但将Group 1 by this text和Group 2 by another text...

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

它应适用于:

anything(\d+)more_text 以及您可以想象的任何模式。

我要做的就是只替换组,并保留其余的比赛。

Answers:


306

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

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

或使用MatchEvaluator:

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

有点混乱的另一种方法可能是使用后向/向前看:

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


17
我编辑了您的答案以提供更多信息,但是您所说的完全正确。不知道我多么想念我可以将所有内容放入组中,无论是否使用它们 :)。我认为,该解决方案比使用先行和后行更好,更干净。
奥斯卡·梅德罗斯

小错字,您的替换方式应为$ 1AA $ 3
Myster

1
为了使这一工作,我不得不添加.Valuem.Groups[1]
jbeldock

10
同样值得注意的是-如果您的替换文本以数字开头,则第一个解决方案(“ $ 1AA $ 3”)将无法正常使用!
Bertie 2014年

2
@OscarMederos,您也可以使用非捕获组-适用于不使用的组。中(?:foo)(bar)$1将替换bar更多详细信息
Patrick


19

我也有此需要,并为此创建了以下扩展方法:

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();
            });
    }
}

用法:

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");

结果:

[assembly: AssemblyFileVersion("1.2.3")][assembly: AssemblyFileVersion("1.2.3")]

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;

请注意,这是假设条件,并且仅在首次出现比赛时有效。
Bartosz

5

这是另一个不错的干净选项,不需要更改模式。

        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));
        });

0

请通过以下代码获取单独的组替换。

new_bib = Regex.Replace(new_bib, @"(?s)(\\bibitem\[[^\]]+\]\{" + pat4 + @"\})[\s\n\v]*([\\\{\}a-zA-Z\.\s\,\;\\\#\\\$\\\%\\\&\*\@\\\!\\\^+\-\\\=\\\~\\\:\\\" + dblqt + @"\\\;\\\`\\\']{20,70})", delegate(Match mts)
                    {
                           var fg = mts.Groups[0].Value.ToString(); 
                           var fs = mts.Groups[1].Value.ToString();
                           var fss = mts.Groups[2].Value.ToString();
                               fss = Regex.Replace(fss, @"[\\\{\}\\\#\\\$\\\%\\\&\*\@\\\!\\\^+\-\\\=\\\~\\\:\\\" + dblqt + @"\\\;\\\`\\\']+", "");
                           return "<augroup>" + fss + "</augroup>" + fs;
                    }, RegexOptions.IgnoreCase);
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.