Пользовательский класс используется в качестве ключа в словаре, но ключ не найден - PullRequest
9 голосов
/ 27 октября 2009

У меня есть класс, показанный ниже, который используется в качестве ключа в Dictionary<ValuesAandB, string> У меня возникают проблемы при попытке найти какой-либо ключ в этом словаре, он никогда не находит его вообще. Как видите, я переопределил Equals и GetHashCode.

Чтобы найти ключ, который я использую

ValuesAandB key = new ValuesAandB(A,B);
if (DictionaryName.ContainsKey(key)) {
   ...
}

Есть что-то еще, что я пропускаю? Кто-нибудь может указать, что я делаю не так?

private class ValuesAandB {
   public string valueA;
   public string valueB;

   // Constructor
   public ValuesAandB (string valueAIn, string valueBIn) {
     valueA = valueAIn;
     valueB = ValueBIn;
   }

   public class EqualityComparer : IEqualityComparer<ValuesAandB> {
      public bool Equals(ValuesAandB x, ValuesAandB y) {
         return ((x.valueA.Equals(y.valueA)) && (x.valueB.Equals(y.valueB)));
      }
      public int GetHashCode(ValuesAandB x) {
         return x.valueA.GetHashCode() ^ x.valueB.GetHashCode();
      }
   }
}

И прежде чем кто-нибудь спросит, да, значения в Словаре!

Ответы [ 4 ]

9 голосов
/ 27 октября 2009

Как вы строите словарь? Вы передаете свой собственный компаратор равенства в свой конструктор?

6 голосов
/ 27 октября 2009

Вы не переопределили Equals и GetHashCode. Вы реализовали второй класс, который может служить в качестве EqualityComparer. Если вы не создадите словарь с помощью EqualityComparer, он не будет использоваться.

Самым простым решением было бы переопределить GetHashCode и Equals напрямую, а не реализовывать компаратор (компараторы, как правило, интересны только тогда, когда вам нужно предоставить несколько различных типов сравнения (например, с учетом регистра и без учета регистра) или когда вам нужно иметь возможность проводить сравнения с классом, который вы не контролируете.

0 голосов
/ 27 октября 2009

У меня была эта проблема, оказывается, словарь сравнивал ссылки для моего ключа, а не значения в объекте.

Я использовал собственный класс Point в качестве ключей. Я переопределил методы ToString () и GetHashCode () и альт, поиск ключей работал нормально.

0 голосов
/ 27 октября 2009

Похоже, вы сравниваете две строки. Iirc, при использовании .Equals () вы сравниваете ссылку на строки, а не фактическое содержимое. Чтобы реализовать EqualityComparer, который работает со строками, вы должны использовать метод String.Compare ().

public class EqualityComparer : IEqualityComparer<ValuesAandB>
{
     public bool Equals(ValuesAandB x, ValuesAandB y)
     {
          return ((String.Compare(x.valueA,y.valueA) == 0) &&
            (String.Compare(x.valueB, y.valueB) == 0));
     }
     // gethashcode stuff here
}

Я мог бы быть немного не в порядке с кодом, который должен приблизить вас ...

...