使用了2个正则表达式。第一个匹配一般的形式,第二个处理内部管道。
对于XML编码,我使用了在System.Security中找到的一个不起眼的小方法。
SecurityElement.Escape Method
. 我在下面的代码中对它进行了充分的限定,以供强调。另一个选择是使用
HttpUtility.HtmlEncode method
但这可能涉及到对System.Web的引用,这取决于您使用它的位置。
string[] inputs = { @"[% 'test' <mtd:ddl id=""asdf"" runat=""server""/> & <%= Integer.MaxValue %> %]",
@"[% 'test' <mtd:ddl id=""asdf"" runat=""server""/> & <%=Integer.MaxValue %> %]",
@"[% 'test' <mtd:ddl id=""asdf"" runat=""server""/> & <%# Integer.MaxValue%> %]",
@"[% 'test' <mtd:ddl id=""asdf"" runat=""server""/> & <%#Integer.MaxValue%> %]",
};
string pattern = @"(?<open>\[%)(?<content>.*?)(?<close>%])";
string expressionPattern = @"(?<content>.*?)(?<tag><%(?:[=#]))\s*(?<expression>.*?)\s*%>";
foreach (string input in inputs)
{
string result = Regex.Replace(input, pattern, m =>
m.Groups["open"].Value +
Regex.Replace(m.Groups["content"].Value, expressionPattern,
expressionMatch =>
System.Security.SecurityElement.Escape(expressionMatch.Groups["content"].Value) +
expressionMatch.Groups["tag"].Value + " " +
expressionMatch.Groups["expression"].Value +
" %>"
) +
m.Groups["close"].Value
);
Console.WriteLine("Before: {0}", input);
Console.WriteLine("After: {0}", result);
}
结果:
Before: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%= Integer.MaxValue %> %]
After: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%= Integer.MaxValue %> %]
Before: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%=Integer.MaxValue %> %]
After: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%= Integer.MaxValue %> %]
Before: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%# Integer.MaxValue%> %]
After: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%# Integer.MaxValue %> %]
Before: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%#Integer.MaxValue%> %]
After: [% 'test' <mtd:ddl id="asdf" runat="server"/> & <%# Integer.MaxValue %> %]
编辑:
如果您不想在最终结果中保留开/关[%%],则将模式更改为:
string pattern = @"\[%(?<content>.*?)%]";
然后确保删除对
m.Groups["open"].Value
和
m.Groups["close"].Value
.