O padrão para KeyValuePair


392

Eu tenho um objeto do tipo IEnumerable<KeyValuePair<T,U>> keyValueList, estou usando

 var getResult= keyValueList.SingleOrDefault();
 if(getResult==/*default */)
 {
 }
 else
 {
 } 

Como posso verificar se getResulté o padrão, caso não encontre o elemento correto?

Não posso verificar se é nullou não, porque KeyValuePairé uma estrutura.


11
Aqui está uma pergunta semelhante com algumas respostas interessantes: stackoverflow.com/q/793897/323071
Andre Luus

2
Você também pode dizer getResult.Key = null para verificar se a lista é nulo como se fosse a chave será sempre nulo!
pengibot

3
@pengibot Isso assume que a chave é anulável.
Justin Skiles

Respostas:


542

Tente o seguinte:

if (getResult.Equals(new KeyValuePair<T,U>()))

ou isto:

if (getResult.Equals(default(KeyValuePair<T,U>)))

5
Você pode usar o conceito da opção "novo" aqui, mas evite especificar novamente o tipo: if (getResult.Equals (Activator.CreateInstance (getResult.GetType ())))
kevinpo

3
Eu acho que if (getResult.Equals(default))também funciona
Simon

2
@ Simon não, não. defaulté igual a nulo. E default(KeyValuePair<T,U>)é um KeyValuePair real que contém null, null. Eu experimentei isso sozinho.
Yokovaski

100

Você pode criar um método de extensão geral (e genérico), como este:

public static class Extensions
{
    public static bool IsDefault<T>(this T value) where T : struct
    {
        bool isDefault = value.Equals(default(T));

        return isDefault;
    }
}

Uso:

// We have to set explicit default value '0' to avoid build error:
// Use of unassigned local variable 'intValue'
int intValue = 0;
long longValue = 12;
KeyValuePair<String, int> kvp1 = new KeyValuePair<String, int>("string", 11);
KeyValuePair<String, int> kvp2 = new KeyValuePair<String, int>();
List<KeyValuePair<String, int>> kvps = new List<KeyValuePair<String, int>> { kvp1, kvp2 };
KeyValuePair<String, int> kvp3 = kvps.FirstOrDefault(kvp => kvp.Value == 11);
KeyValuePair<String, int> kvp4 = kvps.FirstOrDefault(kvp => kvp.Value == 15);

Console.WriteLine(intValue.IsDefault()); // results 'True'
Console.WriteLine(longValue.IsDefault()); // results 'False'
Console.WriteLine(kvp1.IsDefault()); // results 'False'
Console.WriteLine(kvp2.IsDefault()); // results 'True'
Console.WriteLine(kvp3.IsDefault()); // results 'False'
Console.WriteLine(kvp4.IsDefault()); // results 'True'

27

Tente o seguinte:

KeyValuePair<string,int> current = this.recent.SingleOrDefault(r => r.Key.Equals(dialog.FileName) == true);

if (current.Key == null)
    this.recent.Add(new KeyValuePair<string,int>(dialog.FileName,0));

3
Obviamente, isso só funciona se você nunca adicionar acidentalmente ou intencionalmente nulo como chave. Mas eu diria que isso é verdade na grande maioria das vezes, é uma boa ideia. Essa é definitivamente uma verificação muito mais simples do que a maioria das verificações aqui.
NEMEMEM

3
Contanto que a chave seja anulável.
Uri Abramson

10
if(getResult.Key.Equals(default(T)) && getResult.Value.Equals(default(U)))

A única coisa é que isso não é compilado. Os iguais devem ser usados.
Tamas Ionut

8

Eu recomendo uma maneira mais compreensiva usando o método de extensão:

public static class KeyValuePairExtensions
{
    public static bool IsNull<T, TU>(this KeyValuePair<T, TU> pair)
    {
        return pair.Equals(new KeyValuePair<T, TU>());
    }
}

E então basta usar:

var countries = new Dictionary<string, string>
{
    {"cz", "prague"},
    {"de", "berlin"}
};

var country = countries.FirstOrDefault(x => x.Key == "en");

if(country.IsNull()){

}

7

No seu código original, parece que você deseja verificar se a lista estava vazia:

var getResult= keyValueList.SingleOrDefault();
if (keyValueList.Count == 0)
{
   /* default */
}
else
{
}

Você pode simplesmente omitir a primeira linha, pois ela não é usada.
HeyJude

0

Para evitar o boxe, KeyValuePair.Equals(object)você pode usar a ValueTuple.

if ((getResult.Key, getResult.Value) == default)
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.