拆分列表中的字符串值<>

时间:2013-02-14 12:42:02

标签: c# list split

我在字典中包含一个包含以下值的列表

value a
value b
value c
value 1, value 2, value3
value d

我想要的最终结果是

value a
value b
value c
value 1
value 2
value 3
value d

问题是遍历字典并尝试更改集合将无法正常工作,因为我试图在循环时修改它

string[] ar;
        foreach (var kvp in dict)
        {
            if (kvp.Key == "STAR-016")
            {
                foreach (var v in kvp.Value)
                {
                    if (v.Contains(','))
                    {
                        ar = v.Split(',');
                        foreach (var a in ar)
                        {
                            kvp.Value.Add(a);
                        }
                    }
                }
            }
        }

我怎样才能得到理想的结果?

5 个答案:

答案 0 :(得分:2)

您可以使用List代替for循环浏览foreach并修改项目

for (int i = 0; i < kvp.Value.Count; i++)
{
    if (kvp.Value[i].Contains(','))
        ...
}

答案 1 :(得分:2)

假设这只是一个示例,您实际上不仅希望键STAR-016,而且希望值(List<string>)包含逗号的所有内容:

Dictionary<string, List<String>> dict = new Dictionary<string, List<String>>();
dict.Add("STAR-016", new List<string>() { 
    "value a", "value b", "value c", "value 1, value 2, value 3", "value d"
});

foreach (var kvp in dict)
{
    for (int i = kvp.Value.Count - 1; i >= 0; i--)
    {
        string str = kvp.Value[i];
        if (str.Contains(','))
        {
            var parts = str.Split(',').Select(p => p.Trim());
            kvp.Value.RemoveAt(i);
            kvp.Value.InsertRange(i, parts);
        }
    }
}

Demo

我从结束到开始循环以避免并发症,因为InsertRange会添加新字符串,从而增加Count
我使用RemoveAt用逗号替换字符串,并添加新字符串(每个部分用逗号分隔一个)。我使用InsertRange代替AddRange,因为您想保留订单。

结果:

value a
value b
value c
value 1
value 2
value 3
value d

答案 2 :(得分:2)

尝试使用LINQ:

var list = new List<string>()
{
    "value a",
    "value b",
    "value 1, value 2, value 3",
    "value c"
};

/* THIS IS THE IMPORTANT PART: */
/* Replace list with kvp.Value */
list = list.SelectMany( 
         i => i.Split( ',' ).Select( v => v.Trim() ) 
       ).ToList();

foreach ( var item in list )
    Console.WriteLine( item );

输出:

  

值a
  价值b
  值1
  价值2
  价值3
  价值c

要在您的代码中使用它:

foreach (var kvp in dict)
{
    if (kvp.Key == "STAR-016")
    {
        var newList = 
            kvp.Value.SelectMany(
                i => i.Split( ',' ).Select( v => v.Trim() )
            );
        kvp.Value.Clear();
        kvp.Value.AddRange(newList);
    }
}

感谢@Mudu指出更简单的i =>语法

答案 3 :(得分:2)

您应该可以使用SelectMany一行来完成此操作。

dict["STAR-016"] = dict["STAR-016"].SelectMany(s=>s.Split(',')).ToList();

这会将您的密钥列表替换为在逗号上拆分字符串的列表,如果字符串不包含逗号,则只返回字符串。如果您不希望连续逗号产生空字符串,也可以考虑使用StringSplitOptions.RemoveEmptyEntries

答案 4 :(得分:1)

(编辑:juharr的https://stackoverflow.com/a/14875503/17713与内置LINQ功能和更多功能风格基本相同,在描述实际问题时通常更具表现力

我会选择使用yield 不修改原始集合的方法。以下是在List<string> orig上运行的示例代码,该代码也可能在字典中:

public static void Main()
{
    List<string> orig = new List<string>()
    {
        "value a",
        "value b",
        "value c",
        "value 1, value 2, value 3",
        "value d"
    };

    var result = Flatten(orig).ToList();

    foreach(string s in result)
    {
        Console.WriteLine(s);
    }
}

private static IEnumerable<string> Flatten(IList<string> orig)
{
    foreach(string s in orig)
    {
        // split anyway, if there's no colon you just get a one-element
        // array containing s, see
        // http://msdn.microsoft.com/en-us/library/b873y76a.aspx
        foreach(string v in s.Split(','))
        {
            yield return v.Trim();
        }
    }
}

在字典中,您可以将结果替换为原始值:

dict["STAR-016"] = Flatten(dict["STAR-016"]).ToList()

附注:上面的代码直接找到STAR-016,而不是使用速度较慢的foreach。如果你没有缩短代码并且实际上只是查找STAR-016,我建议你使用这种字典查找方式。