Преобразуйте список в словарь, используя linq и не заботясь о дубликатах - PullRequest
133 голосов
/ 23 июля 2010

У меня есть список объектов Person.Я хочу преобразовать в словарь, где ключ - это имя и фамилия (объединенные), а значение - объект Person.

Проблема в том, что у меня есть несколько дублированных людей, поэтому при использованииэтот код:

private Dictionary<string, Person> _people = new Dictionary<string, Person>();

_people = personList.ToDictionary(
    e => e.FirstandLastName,
    StringComparer.OrdinalIgnoreCase);

Я знаю, это звучит странно, но сейчас мне не нужны дубликаты имен.Если есть несколько имен, я просто хочу получить одно.Могу ли я в любом случае написать этот код выше, чтобы он занимал одно из имен и не дублировал дубликаты?

Ответы [ 10 ]

376 голосов
/ 23 июля 2010

LINQ решение:

// Use the first value in group
var _people = personList
    .GroupBy(p => p.FirstandLastName, StringComparer.OrdinalIgnoreCase)
    .ToDictionary(g => g.Key, g => g.First(), StringComparer.OrdinalIgnoreCase);

// Use the last value in group
var _people = personList
    .GroupBy(p => p.FirstandLastName, StringComparer.OrdinalIgnoreCase)
    .ToDictionary(g => g.Key, g => g.Last(), StringComparer.OrdinalIgnoreCase);

Если вы предпочитаете не-LINQ решение, вы можете сделать что-то вроде этого:

// Use the first value in list
var _people = new Dictionary<string, Person>(StringComparer.OrdinalIgnoreCase);
foreach (var p in personList)
{
    if (!_people.ContainsKey(p.FirstandLastName))
        _people[p.FirstandLastName] = p;
}

// Use the last value in list
var _people = new Dictionary<string, Person>(StringComparer.OrdinalIgnoreCase);
foreach (var p in personList)
{
    _people[p.FirstandLastName] = p;
}
58 голосов
/ 23 июля 2010

Вот очевидное решение, отличное от linq:

foreach(var person in personList)
{
  if(!myDictionary.Keys.Contains(person.FirstAndLastName))
    myDictionary.Add(person.FirstAndLastName, person);
}
37 голосов
/ 13 апреля 2011

Решение Linq с использованием Distinct () и без группировки:

var _people = personList
    .Select(item => new { Key = item.Key, FirstAndLastName = item.FirstAndLastName })
    .Distinct()
    .ToDictionary(item => item.Key, item => item.FirstFirstAndLastName, StringComparer.OrdinalIgnoreCase);

Я не знаю, является ли оно более хорошим, чем решение LukeH, но оно также работает.

27 голосов
/ 31 августа 2011

Это должно работать с лямбда-выражением:

personList.Distinct().ToDictionary(i => i.FirstandLastName, i => i);
10 голосов
/ 11 июня 2014

Вы также можете использовать функцию ToLookup LINQ, которую вы затем можете использовать почти взаимозаменяемо со словарем.

_people = personList
    .ToLookup(e => e.FirstandLastName, StringComparer.OrdinalIgnoreCase);
_people.ToDictionary(kl => kl.Key, kl => kl.First()); // Potentially unnecessary

Это, в сущности, сделает GroupBy в ответе Люка * , но даст хеширование, которое обеспечивает словарь. Таким образом, вам, вероятно, не нужно преобразовывать его в словарь, а просто использовать функцию LINQ First всякий раз, когда вам нужно получить доступ к значению ключа.

7 голосов
/ 23 июля 2010

Чтобы обработать удаление дубликатов, реализуйте IEqualityComparer<Person>, который можно использовать в методе Distinct(), и тогда получить ваш словарь будет легко. Дано:

class PersonComparer : IEqualityComparer<Person>
{
    public bool Equals(Person x, Person y)
    {
        return x.FirstAndLastName.Equals(y.FirstAndLastName, StringComparison.OrdinalIgnoreCase);
    }

    public int GetHashCode(Person obj)
    {
        return obj.FirstAndLastName.ToUpper().GetHashCode();
    }
}

class Person
{
    public string FirstAndLastName { get; set; }
}

Получите ваш словарь:

List<Person> people = new List<Person>()
{
    new Person() { FirstAndLastName = "Bob Sanders" },
    new Person() { FirstAndLastName = "Bob Sanders" },
    new Person() { FirstAndLastName = "Jane Thomas" }
};

Dictionary<string, Person> dictionary =
    people.Distinct(new PersonComparer()).ToDictionary(p => p.FirstAndLastName, p => p);
5 голосов
/ 19 марта 2014

Вы можете создать метод расширения, аналогичный ToDictionary (), с той разницей, что он допускает дублирование. Что-то вроде:

    public static Dictionary<TKey, TElement> SafeToDictionary<TSource, TKey, TElement>(
        this IEnumerable<TSource> source, 
        Func<TSource, TKey> keySelector, 
        Func<TSource, TElement> elementSelector, 
        IEqualityComparer<TKey> comparer = null)
    {
        var dictionary = new Dictionary<TKey, TElement>(comparer);

        if (source == null)
        {
            return dictionary;
        }

        foreach (TSource element in source)
        {
            dictionary[keySelector(element)] = elementSelector(element);
        }

        return dictionary; 
    }

В этом случае, если есть дубликаты, выигрывает последнее значение.

2 голосов
/ 15 февраля 2019

В случае, если мы хотим, чтобы все люди (а не только один человек) в возвращаемом словаре, мы могли бы:

var _people = personList
.GroupBy(p => p.FirstandLastName)
.ToDictionary(g => g.Key, g => g.Select(x=>x));
2 голосов
/ 27 апреля 2016
        DataTable DT = new DataTable();
        DT.Columns.Add("first", typeof(string));
        DT.Columns.Add("second", typeof(string));

        DT.Rows.Add("ss", "test1");
        DT.Rows.Add("sss", "test2");
        DT.Rows.Add("sys", "test3");
        DT.Rows.Add("ss", "test4");
        DT.Rows.Add("ss", "test5");
        DT.Rows.Add("sts", "test6");

        var dr = DT.AsEnumerable().GroupBy(S => S.Field<string>("first")).Select(S => S.First()).
            Select(S => new KeyValuePair<string, string>(S.Field<string>("first"), S.Field<string>("second"))).
           ToDictionary(S => S.Key, T => T.Value);

        foreach (var item in dr)
        {
            Console.WriteLine(item.Key + "-" + item.Value);
        }
0 голосов
/ 23 октября 2015

Начиная с решения Карры, вы также можете написать его как:

foreach(var person in personList.Where(el => !myDictionary.ContainsKey(el.FirstAndLastName)))
{
    myDictionary.Add(person.FirstAndLastName, person);
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...