Digite relacionamentos em operações de consulta LINQ (C#)
Para escrever consultas de forma eficaz, você deve entender como os tipos de variáveis em uma operação de consulta completa se relacionam entre si. Se você entender essas relações, compreenderá mais facilmente os exemplos de código e exemplos de código do LINQ na documentação. Além disso, você entenderá o que ocorre quando as variáveis são digitadas implicitamente usando var
.
As operações de consulta LINQ são fortemente tipadas na fonte de dados, na própria consulta e na execução da consulta. O tipo das variáveis na consulta deve ser compatível com o tipo dos elementos na fonte de dados e com o tipo da variável de iteração na foreach
instrução. Essa digitação forte garante que os erros de tipo sejam detetados no momento da compilação, quando podem ser corrigidos antes que os usuários os encontrem.
Para demonstrar essas relações de tipo, a maioria dos exemplos a seguir usa digitação explícita para todas as variáveis. O último exemplo mostra como os mesmos princípios se aplicam mesmo quando você usa digitação implícita usando var
.
Consultas que não transformam os dados de origem
A ilustração a seguir mostra uma operação de consulta LINQ to Objects que não executa transformações nos dados. A origem contém uma sequência de cadeias de caracteres e a saída da consulta também é uma sequência de cadeias de caracteres.
- O argumento type da fonte de dados determina o tipo da variável de intervalo.
- O tipo do objeto selecionado determina o tipo da variável de consulta. Aqui
name
está uma string. Portanto, a variável de consulta é umIEnumerable<string>
arquivo . - A variável de consulta é iterada
foreach
na instrução. Como a variável de consulta é uma sequência de cadeias de caracteres, a variável de iteração também é uma cadeia de caracteres.
Consultas que transformam os dados de origem
A ilustração a seguir mostra uma operação de consulta LINQ to SQL que executa uma transformação simples nos dados. A consulta usa uma sequência de Customer
objetos como entrada e seleciona apenas a Name
propriedade no resultado. Como Name
é uma cadeia de caracteres, a consulta produz uma sequência de cadeias de caracteres como saída.
- O argumento type da fonte de dados determina o tipo da variável de intervalo.
- A
select
instrução retorna aName
propriedade em vez do objeto completoCustomer
. ComoName
é uma cadeia de caracteres, o argumento type decustNameQuery
éstring
, nãoCustomer
. - Como
custNameQuery
é uma sequência de cadeias de caracteres, aforeach
variável de iteração do loop também deve ser umstring
arquivo .
A ilustração a seguir mostra uma transformação um pouco mais complexa. A select
instrução retorna um tipo anônimo que captura apenas dois membros do objeto original Customer
.
- O argumento type da fonte de dados é sempre o tipo da variável range na consulta.
- Como a
select
instrução produz um tipo anônimo, a variável de consulta deve ser digitada implicitamente usandovar
. - Como o tipo da variável de consulta está implícito, a variável de iteração no
foreach
loop também deve estar implícita.
Permitindo que o compilador infera informações de tipo
Embora você deva entender as relações de tipo em uma operação de consulta, você tem a opção de permitir que o compilador faça todo o trabalho para você. A palavra-chave var pode ser usada para qualquer variável local em uma operação de consulta. A ilustração a seguir é semelhante ao exemplo número 2 que foi discutido anteriormente. No entanto, o compilador fornece o tipo forte para cada variável na operação de consulta.
LINQ e tipos genéricos (C#)
As consultas LINQ são baseadas em tipos genéricos. Você não precisa de um conhecimento aprofundado de genéricos antes de começar a escrever consultas. No entanto, você pode querer entender dois conceitos básicos:
- Ao criar uma instância de uma classe de coleção genérica, como List<T>, você substitui o "T" pelo tipo de objetos que a lista conterá. Por exemplo, uma lista de cadeias de caracteres é expressa como
List<string>
, e uma lista deCustomer
objetos é expressa comoList<Customer>
. Uma lista genérica é fortemente tipada e oferece muitos benefícios em relação às coleções que armazenam seus elementos como Object. Se você tentar adicionar umCustomer
a umList<string>
, você receberá um erro em tempo de compilação. É fácil usar coleções genéricas porque você não precisa executar a conversão de tipo em tempo de execução. - IEnumerable<T> é a interface que permite que classes de coleção genéricas sejam enumeradas usando a
foreach
instrução. As classes de coleção genéricas suportam IEnumerable<T> exatamente como as classes de coleção não genéricas, como ArrayList o suporte IEnumerable.
Para obter mais informações sobre genéricos, consulte Genéricos.
IEnumerable<T> variáveis em consultas LINQ
As variáveis de consulta LINQ são digitadas como IEnumerable<T> ou um tipo derivado, como IQueryable<T>. Quando você vê uma variável de consulta que é digitada como IEnumerable<Customer>
, isso significa apenas que a consulta, quando é executada, produzirá uma sequência de zero ou mais Customer
objetos.
IEnumerable<Customer> customerQuery =
from cust in customers
where cust.City == "London"
select cust;
foreach (Customer customer in customerQuery)
{
Console.WriteLine($"{customer.LastName}, {customer.FirstName}");
}
Permitindo que o compilador manipule declarações de tipo genéricas
Se preferir, você pode evitar a sintaxe genérica usando a palavra-chave var . A var
palavra-chave instrui o compilador a inferir o tipo de uma variável de consulta examinando a fonte de dados especificada na from
cláusula. O exemplo a seguir produz o mesmo código compilado que o exemplo anterior:
var customerQuery2 =
from cust in customers
where cust.City == "London"
select cust;
foreach(var customer in customerQuery2)
{
Console.WriteLine($"{customer.LastName}, {customer.FirstName}");
}
A var
palavra-chave é útil quando o tipo da variável é óbvio ou quando não é tão importante especificar explicitamente tipos genéricos aninhados, como aqueles que são produzidos por consultas de grupo. Em geral, recomendamos que, se você usar var
o , perceba que isso pode tornar seu código mais difícil para outras pessoas lerem. Para obter mais informações, consulte Variáveis locais digitadas implicitamente.