我怎么能将这些foreach循环转换为LINQ表达式?

时间:2014-10-25 21:44:47

标签: c# linq

我使用ReSharper检查项目中的代码问题,并通知我以下循环可以转换为LINQ表达式:

var dictionary = new Dictionary<string, string[]>
{
    { "400", new[] { "12345", "54321", "51423" } },
    { "500", new[] { "67890", "09876", "63727" } },
    { "600", new[] { "41713", "98234", "96547" } },
    { "700", new[] { "00000", "67990", "83752" } }
};

// ...

var targetValue = "41713";
foreach (string group in dictionary.Keys)
{
    foreach (string name in dictionary[group])
    {
        if (name == targetValue)
            return group;
    }
}
return "User";

循环基本上检查字典的值(字符串数组)以查看targetValue是否属于它们中的任何一个并返回该数组的密钥(如果在内部找到)。

我尝试了以下操作,但很明显,只要其值等于targetValue,它就会返回内部值。

var r = dictionary
        .SelectMany(t => t.Value)
        .FirstOrDefault(t => t == targetValue);

3 个答案:

答案 0 :(得分:4)

所以你想获得字典中第一个string[] - 值包含给定值的键吗?

var pairs = dictionary.Where(kv => kv.Value.Contains(myValue));
if (pairs.Any())
{
    string group = pairs.First().Key;
}

或者可读性较低但效率稍高,因为它只执行一次查询:

var pair = dictionary.FirstOrDefault(kv => kv.Value.Contains(myValue));
if (!pair.Equals(default(KeyValuePair<string, string[]>)))
{
    string group = pair.Key;
}

最后但并非最不重要的另一种方法是我最喜欢的,也使用"User" - 默认:

string group = dictionary.Where(kv => kv.Value.Contains(myValue))
    .Select(kv=> kv.Key)
    .DefaultIfEmpty("User")
    .First();

答案 1 :(得分:1)

var r = dictionary.FirstOrDefault(
    x => x.Value.FirstOrDefault(y => y == myValue) != null);

答案 2 :(得分:0)

如果它不存在,这也会得到所需的值或null:

编辑:

var result = dictionary.SkipWhile(n => !n.Value.Contains(myValue)).FirstOrDefault().Key;
//another way to get the key
//var result = dictionary.SingleOrDefault(n => n.Value.Contains(myValue)).Key;
if (result != null)
{
   //do whatever with the result variable here
}