使用C#为字符串创建正则表达式模式

时间:2016-08-18 08:31:01

标签: c# regex

我有这样的字符串模式:

#c1 12,34,222x8. 45,989,100x10. 767x55. #c1

我想将这些模式改为:

c1,12,8
c1,34,8
c1,222,8
c1,45,10
c1,989,10
c1,100,10
c1,767,55

我在C#中的代码:

private void btnProses_Click(object sender, EventArgs e)
{
    String ps = txtpesan.Text;
    Regex rx = new Regex("((?:\d+,)*(?:\d+))x(\d+)");
    Match mc = rx.Match(ps);
    while (mc.Success)
    {
        txtpesan.Text = rx.ToString();
    }
}

我一直在使用splitreplace,但无济于事。在我尝试解决此问题后,我看到很多人使用regex,我尝试使用regex,但我没有得到制作模式正则表达式的逻辑。

我应该用什么来解决这个问题?

2 个答案:

答案 0 :(得分:2)

有时正则表达式不是好方法 - 老派的方式胜出。假设有效输入:

var tokens = txtpesan.Text.Split(' '); //or use split by regex's whitechar
var prefix = tokens[0].Trim('#');

var result = new StringBuilder();

//skip first and last token
foreach (var token in tokens.Skip(1).Reverse().Skip(1).Reverse())
{
    var xIndex = token.IndexOf("x");
    var numbers = token.Substring(0, xIndex).Split(',');
    var lastNumber = token.Substring(xIndex + 1).Trim('.');

    foreach (var num in numbers)
    {
        result.AppendLine(string.Format("{0},{1},{2}", prefix, num, lastNumber));
    }
}

var viola = result.ToString();
Console.WriteLine(viola);

答案 1 :(得分:0)

这是一个有点难看的基于正则表达式的解决方案:

var q = "#c1 12,34,222x8. 45,989,100x10. 767x55. #c1";
var results = Regex.Matches(q, @"(?:(?:,?\b(\d+))(?:x(\d+))?)+");
var caps = results.Cast<Match>()
     .Select(m => m.Groups[1].Captures.Cast<Capture>().Select(cap => cap.Value));
var trailings = results.Cast<Match>().Select(m => m.Groups[2].Value).ToList();
var c1 = q.Split(' ')[0].Substring(1);
var cnt = 0;
foreach (var grp in caps)
{
     foreach (var item in grp)
     {
         Console.WriteLine("{0},{1},{2}", c1, item, trailings[cnt]);
     }
     cnt++;
}

regex demo can be seen here。该模式匹配以逗号分隔的数字块,同时将数字捕获到组1中,并将x后的数字捕获到组2.无法摆脱cnt计数器,抱歉。