18

これらの2つの列挙型フィールドを持つこの単純なクラスがあります。コレクション(List<T>)でこのオブジェクトの1つのアイテムを見つけようとしていますが、Containsメソッドが正しく機能しません。

public class Calculator : IEqualityComparer<Calculator>
{
    public DashboardsComputationMode ComputationMode { get; set; }
    public Modes Mode { get; set; }

    public Calculator(DashboardsComputationMode dashboardsComputationMode, Modes mode)
    {
        ComputationMode = dashboardsComputationMode;
        Mode = mode;
    }

    public bool Equals(Calculator x, Calculator y)
    {
        return (x.ComputationMode.Equals(y.ComputationMode) && x.Mode.Equals(y.Mode));
    }

    public int GetHashCode(Calculator obj)
    {
        return obj.ComputationMode.GetHashCode() ^ obj.Mode.GetHashCode();
    }
}

public enum DashboardsComputationMode
{
    Weighted = 0,
    Aggregated = 1,
    PR = 2,
    CurrentValue = 3,
    EquivalentHours = 4,
    AggregatedCorrected = 5,
    PRCorrected = 6
}

public enum Modes
{
    InstantaneousMode = 0,
    DailyMode = 1,
    MonthlyMode = 2,
    YearlyMode = 5,
    Undefined = 4,
}

このテストが機能しないのはなぜですか

[TestMethod]
public void TestMethod1()
{
  var list = new List<Calculator>()
  {
    new Calculator(DashboardsComputationMode.PR, Modes.DailyMode),
    new Calculator(DashboardsComputationMode.CurrentValue, Modes.YearlyMode),
    new Calculator(DashboardsComputationMode.PRCorrected, Modes.MonthlyMode)
  };

  var item = new Calculator(DashboardsComputationMode.CurrentValue, Modes.YearlyMode);
  Assert.IsTrue(list[1].Equals(item));
  Assert.IsTrue(list.Contains(item));
}

最初のアサートは正常に機能します

Assert.IsTrue(list[1].Equals(item)) 

しかし、2番目はしません

Assert.IsTrue(list.Contains(item));
4

2 に答える 2

24

List<T>.Containsは、デフォルトの等値比較子 ( によって返されるものEqualityComparer<T>.Default) を使用して等値を判定します。

仕組みに関するMSDNの説明は次のEqualityComparer<T>.Defaultとおりです。

Default プロパティは、型 T が System.IEquatableインターフェイスを実装しているかどうかを確認し、実装している場合は、その実装を使用する EqualityComparerを返します。それ以外の場合は、T によって提供されるObject.EqualsおよびObject.GetHashCodeのオーバーライドを使用 するEqualityComparerを返します。

つまり、CalculatorクラスはSystem.IEquatable ( !ではなくSystem.IEqualityComparer) インターフェイスを実装するか、 Object.EqualsおよびObject.GetHashCodeメソッドをオーバーライドする必要があります。

于 2013-01-29T10:45:15.320 に答える
5

IEqualityComparer<Calculator>Equals と Contains の両方で使用していません。EqualityComparer意味が違います。コードを修正しました。

  public class CalculatorComparer : IEqualityComparer<Calculator>
{

    public bool Equals(Calculator x, Calculator y)
    {
        return (x.ComputationMode.Equals(y.ComputationMode) && x.Mode.Equals(y.Mode));
    }

    public int GetHashCode(Calculator obj)
    {
        return obj.ComputationMode.GetHashCode() ^ obj.Mode.GetHashCode();
    }
}
public class Calculator
{
    public DashboardsComputationMode ComputationMode { get; set; }
    public Modes Mode { get; set; }

    public Calculator(DashboardsComputationMode dashboardsComputationMode, Modes mode)
    {
        ComputationMode = dashboardsComputationMode;
        Mode = mode;
    }


    public override bool Equals(object obj)
    {
        Calculator y = obj as Calculator;

        return (this.ComputationMode.Equals(y.ComputationMode) && this.Mode.Equals(y.Mode));
    }
}

public enum DashboardsComputationMode
{
    Weighted = 0,
    Aggregated = 1,
    PR = 2,
    CurrentValue = 3,
    EquivalentHours = 4,
    AggregatedCorrected = 5,
    PRCorrected = 6
}

public enum Modes
{
    InstantaneousMode = 0,
    DailyMode = 1,
    MonthlyMode = 2,
    YearlyMode = 5,
    Undefined = 4,
}

これで、両方が true を返すはずです。

于 2013-01-29T10:45:01.113 に答える