Como obtenho a linha MAX com GROUP BY na consulta LINQ?

94

Estou procurando uma maneira no LINQ de corresponder à consulta SQL a seguir.

Select max(uid) as uid, Serial_Number from Table Group BY Serial_Number

Realmente estou procurando alguma ajuda sobre este. A consulta acima obtém o uid máximo de cada número de série devido à Group Bysintaxe.

SpoiledTechie.com
fonte

Respostas:

92
        using (DataContext dc = new DataContext())
        {
            var q = from t in dc.TableTests
                    group t by t.SerialNumber
                        into g
                        select new
                        {
                            SerialNumber = g.Key,
                            uid = (from t2 in g select t2.uid).Max()
                        };
        }
Tvanfosson
fonte
1
Quero aceitar as duas respostas, mas acho que não posso, então votei em vocês dois. Muitíssimo obrigado!!!
SpoiledTechie.com
68
var q = from s in db.Serials
        group s by s.Serial_Number into g
        select new {Serial_Number = g.Key, MaxUid = g.Max(s => s.uid) }
DamienG
fonte
Quero aceitar as duas respostas, mas acho que não posso, então votei em vocês dois. Muitíssimo obrigado!!!
SpoiledTechie.com
Observe também que a variável para a expressão => lambda na função Max pode ser qualquer coisa (s => s.uid, tv => tv.uid, asdf => asdf.uid). O Linq o reconhecerá automaticamente como selecionando elementos do tipo Serial.
Michael,
29

Na forma de cadeia de métodos:

db.Serials.GroupBy(i => i.Serial_Number).Select(g => new
    {
        Serial_Number = g.Key,
        uid = g.Max(row => row.uid)
    });
Lu55
fonte
22

Eu verifiquei a resposta de DamienG no LinqPad. Ao invés de

g.Group.Max(s => s.uid)

deveria estar

g.Max(s => s.uid)

Obrigado!

denis_n
fonte
10
Isso provavelmente deve ser um comentário sobre a resposta de DamienG.
Cᴏʀʏ
10

As respostas estão OK se você precisar apenas desses dois campos, mas para um objeto mais complexo, talvez esta abordagem possa ser útil:

from x in db.Serials 
group x by x.Serial_Number into g 
orderby g.Key 
select g.OrderByDescending(z => z.uid)
.FirstOrDefault()

... isso evitará o "selecionar novo"

Javier
fonte
Eu gosto disso - você sabe se é tão eficiente quanto as outras respostas?
noelicus
Você usou a sintaxe de consulta aqui até o final, onde mudou para a sintaxe de método. Você teve que fazer? É o formulário de sintaxe de consulta completa para sua solução?
Seth
0

Isso pode ser feito usando GroupBy e SelectMany na expressão lamda do LINQ

var groupByMax = list.GroupBy(x=>x.item1).SelectMany(y=>y.Where(z=>z.item2 == y.Max(i=>i.item2)));
Abhas Bhoi
fonte