Разница между Select и SelectMany

794

Я искал разницу между Select и SelectMany, но я не смог найти подходящий ответ. Мне нужно узнать разницу при использовании LINQ To SQL, но все, что я нашел, это стандартные примеры массивов.

Может ли кто-нибудь предоставить пример LINQ To SQL?

  • 7
    Вы можете посмотреть на код для SelectMany с одной функцией или с двумя функциями reference. source.microsoft.com/#System.Core/System/Linq/…
  • 1
    Если вы знакомы с Kotlin, у него есть довольно похожие реализации для коллекций, такие как map, также называемая C # Select и flatMap, также называемая C # SelectMany. По сути, функции расширения библиотеки Kotlin std для коллекций похожи на библиотеку C # Linq.
Теги:
linq
linq-to-sql

12 ответов

1281
Лучший ответ

SelectMany выравнивает запросы, которые возвращают списки списков. Например

public class PhoneNumber
{
    public string Number { get; set; }
}

public class Person
{
    public IEnumerable<PhoneNumber> PhoneNumbers { get; set; }
    public string Name { get; set; }
}

IEnumerable<Person> people = new List<Person>();

// Select gets a list of lists of phone numbers
IEnumerable<IEnumerable<PhoneNumber>> phoneLists = people.Select(p => p.PhoneNumbers);

// SelectMany flattens it to just a list of phone numbers.
IEnumerable<PhoneNumber> phoneNumbers = people.SelectMany(p => p.PhoneNumbers);

// And to include data from the parent in the result: 
// pass an expression to the second parameter (resultSelector) in the overload:
var directory = people
   .SelectMany(p => p.PhoneNumbers,
               (parent, child) => new { parent.Name, child.Number });

Live Demo on.NET Fiddle

  • 5
    А что если у человека есть свойство Name, и вы также хотите сохранить его вместе с номерами телефонов?
  • 7
    @David SelectMany (p => new {PhoneNumbers = p.PhoneNumbers, Name = p.Name}) должен приблизить вас.
Показать ещё 5 комментариев
134

Выбирать многие - это как операция перекрестного соединения в SQL, где требуется перекрестный продукт.
Например, если у нас есть

Set A={a,b,c}
Set B={x,y}

Выбирать многие можно, чтобы получить следующий набор

{ (x,a) , (x,b) , (x,c) , (y,a) , (y,b) , (y,c) }

Заметим, что здесь мы берем все возможные комбинации, которые могут быть сделаны из элементов множества A и множества B.

Вот пример LINQ, который вы можете попробовать

List<string> animals = new List<string>() { "cat", "dog", "donkey" };
List<int> number = new List<int>() { 10, 20 };

var mix = number.SelectMany(num => animals, (n, a) => new { n, a });

смесь будет иметь следующие элементы в плоской структуре, такой как

{(10,cat), (10,dog), (10,donkey), (20,cat), (20,dog), (20,donkey)}
  • 2
    Я знаю, что это старо, но я хотел поблагодарить вас за это, это спасло меня очень много! :) Полезно иметь ссылку и на эти коды: stackoverflow.com/questions/3479980/… Cheers!
  • 3
    SelectMany не должен использоваться таким образом. У него есть возможность просто взять одну функцию.
Показать ещё 2 комментария
84

Изображение 1002

var players = db.SoccerTeams.Where( c=> c.Country == "Spain")
                            .SelectMany( c => c.players);

foreach(var player in players)
{
    Console.WriteLine( player.LastName);
}
  1. Роналду
  2. Месси
  3. Фабрегас
  4. тюк
  5. Касильяс

...

  • 0
    отличный пример данных
63

SelectMany() позволяет свернуть многомерную последовательность таким образом, чтобы в противном случае потребовался бы второй Select() или цикл.

Подробнее об этом сообщении в блоге.

  • 0
    Но первый возвращает тип Enumerables для детей, а второй пример возвращает тип Parents? На самом деле, я немного запутался, не могли бы вы открыть его немного больше?
  • 0
    На самом деле, наоборот. Второе полностью сгладит иерархию перечислимых элементов, так что вы вернете детей. Попробуйте статью по ссылке, которую я добавил, посмотрите, поможет ли это.
Показать ещё 3 комментария
32

Есть несколько перегрузок в SelectMany. Один из них позволяет отслеживать любые отношения между родителем и детьми при обходе иерархии.

Пример: предположим, что у вас есть следующая структура: League → Teams → Player.

Вы можете легко вернуть плоскую коллекцию игроков. Однако вы можете потерять любую ссылку на команду, частью которой является игрок.

К счастью, для этой цели существует перегрузка:

var teamsAndTheirLeagues = 
         from helper in leagues.SelectMany
               ( l => l.Teams
                 , ( league, team ) => new { league, team } )
                      where helper.team.Players.Count > 2 
                           && helper.league.Teams.Count < 10
                           select new 
                                  { LeagueID = helper.league.ID
                                    , Team = helper.team 
                                   };

Предыдущий пример взят из блога Dan IK. Я настоятельно рекомендую вам взглянуть на него.

18

Я понимаю SelectMany, чтобы работать как ярлык соединения.

Итак, вы можете:

var orders = customers
             .Where(c => c.CustomerName == "Acme")
             .SelectMany(c => c.Orders);
  • 0
    Приведенный пример работает, но SelectMany не совсем работает как соединение. Объединение позволяет «использовать» любое поле исходной таблицы, а также любое поле объединенной таблицы. Но здесь вы должны указать объект списка, прикрепленный к исходной таблице. Например, .SelectMany(c => new {c.CompanyName, c.Orders.ShippedDate}); не будет работать. SelectMany довольно сглаживает список списков - и вы можете выбрать любой (но только по одному) из содержащихся списков для результата. Для сравнения: внутреннее соединение в Linq .
12

Выбор - это простая взаимно-однозначная проекция от исходного элемента к элементу результата. Выбрать- Многие используются, когда в выражении запроса есть несколько из предложений: каждый элемент в исходной последовательности используется для генерации новой последовательности.

7

Некоторые функции SelectMany могут не понадобиться. Ниже 2 запроса дают тот же результат.

Customers.Where(c=>c.Name=="Tom").SelectMany(c=>c.Orders)

Orders.Where(o=>o.Customer.Name=="Tom")

Для отношений "один-ко-многим"

  • Если "Начать" с "1", требуется SelectMany, он сглаживает многие.
  • Если "Начать" с "Много", SelectMany не требуется. ( все еще можно фильтровать от "1" , это проще, чем ниже стандартного запроса на соединение)

from o in Orders
join c in Customers on o.CustomerID equals c.ID
where c.Name == "Tom"
select o
3

Без получения технической базы данных со многими организациями, каждая со многими пользователями: -

var orgId = "123456789";

var userList1 = db.Organizations
                   .Where(a => a.OrganizationId == orgId)
                   .SelectMany(a => a.Users)
                   .ToList();

var userList2 = db.Users
                   .Where(a => a.OrganizationId == orgId)
                   .ToList();

оба возвращают один и тот же список ApplicationUser для выбранной организации.

Первые "проекты" от "Организации к пользователям", вторая запрашивает таблицу "Пользователи" напрямую.

2

Просто для альтернативного представления, которое может помочь некоторым функциональным программистам:

  • Select map
  • SelectMany bind (или flatMap для ваших людей Scala/Kotlin)
1

Это более ясно, когда запрос возвращает строку (массив символов):

Например, если в списке "Фрукты" содержится "яблоко",

'Select' возвращает строку:

Fruits.Select(s=>s) 

[0]: "apple"

"SelectMany" выравнивает строку:

Fruits.SelectMany(s=>s)

[0]: 97  'a'
[1]: 112 'p'
[2]: 112 'p'
[3]: 108 'l'
[4]: 101 'e'
-4

Это лучший способ понять, что я думаю.

            var query =
            Enumerable
                .Range(1, 10)
                .SelectMany(ints => Enumerable.Range(1, 10), (a, b) => $"{a} * {b} = {a * b}")
                .ToArray();

        Console.WriteLine(string.Join(Environment.NewLine, query));

        Console.Read();

Пример таблицы умножения.

  • 3
    Только если значение «лучший» сильно изменилось.
  • 2
    так что это лучший способ, как ты думаешь ?? тогда каков сложный способ мышления ??

Ещё вопросы

Сообщество Overcoder
Наверх
Меню