使用LINQ创建字典并避免“已添加具有相同键的项”错误

3htmauhk  于 2022-12-06  发布在  其他
关注(0)|答案(2)|浏览(178)

我想在字典中找到一个键,如果找到,替换值,如果没有,添加键/值。
编码:

public class MyObject
{

    public string UniqueKey { get; set; }
    public string Field1 { get; set; }
    public string Field2 { get; set; }

}

LINQ解决方案(抛出An item with the same key has already been added.):

Dictionary<string, MyObject> objectDict = csvEntries.ToDictionary(csvEntry => csvEntry.ToMyObject().UniqueKey, csvEntry => csvEntry.ToMyObject());

ForEach解决方案(工作):

Dictionary<string, MyObject> objectDict = new Dictionary<string, MyObject>();
foreach (CSVEntry csvEntry in csvEntries)
{

    MyObject obj = csvEntry.ToMyObject();

    if (objectDict.ContainsKey(obj.UniqueKey))
    {
        objectDict[obj.UniqueKey] = obj;
    }
    else {
        objectDict.Add(obj.UniqueKey, obj);
    }

}

我真的很喜欢LINQ解决方案,但就目前而言,它抛出了上述错误。有没有什么好的方法来避免错误并使用LINQ?

huwehgph

huwehgph1#

可以使用GroupBy创建唯一键:

Dictionary<string, MyObject> objectDict = csvEntries
    .Select(csvEntry => csvEntry.ToMyObject())
    .GroupBy(x => x.UniqueKey)
    .ToDictionary(grp => grp.Key, grp => grp.First());

然而,你可以用ToListToArray来代替grp.First()创建一个集合,这样你就不会在出现重复键的情况下使用任意的对象,或者在First之前的OrderBy中添加你的优先级逻辑:grp => grp.OrderBy(x => Field1).ThenBy(x => x.Field2).First()
另一个选择是使用Lookup<TKey, TValue>,它允许重复的键,甚至不存在的键,在这种情况下,你会得到一个空序列。

var uniqueKeyLookup = csvEntries
    .Select(csvEntry => csvEntry.ToMyObject())
    .ToLookup(x => x.UniqueKey);
IEnumerable<MyObject> objectsFor1234 = uniqueKeyLookup["1234"]; // empty if it doesn't exist
tquggr8v

tquggr8v2#

在Tim的answer的基础上,可以使用以下扩展方法,这样就不需要在整个项目中重复实现:

public static class DictionaryExtensions
{
    public static Dictionary<TKey, TValue> ToDictionaryWithDupSelector<TKey, TValue>(
        this IEnumerable<TValue> enumerable,
        Func<TValue, TKey> groupBy, Func<IEnumerable<TValue>, TValue> selector = null) {

        if (selector == null)
            selector = new Func<IEnumerable<TValue>, TValue>(grp => grp.First());

        return enumerable
            .GroupBy(e => groupBy(e))
            .ToDictionary(grp => grp.Key, grp => selector(grp));
    }
}

默认情况下,当有重复的元素时,它将选择第一个元素,但我提供了一个可选参数,您可以在其中指定一个替代选择器。

var objList = new List<string[]> {
    new string[2] {"1", "first"},
    new string[2] {"1", "last"},
    new string[2] {"2", "you"},
};
var asDict = objList.ToDictionary(
    arr => arr[0],
    grp => grp.Last()
);

相关问题