2009-10-25 46 views
8

我有以下類:LinQ在不同的使用自定義比較葉重複

public class SupplierCategory : IEquatable<SupplierCategory> 
{ 
    public string Name { get; set; } 
    public string Parent { get; set; } 

    #region IEquatable<SupplierCategory> Members 

    public bool Equals(SupplierCategory other) 
    { 
     return this.Name == other.Name && this.Parent == other.Parent; 
    } 

    #endregion 
} 

public class CategoryPathComparer : IEqualityComparer<List<SupplierCategory>> 
{ 
    #region IEqualityComparer<List<SupplierCategory>> Members 

    public bool Equals(List<SupplierCategory> x, List<SupplierCategory> y) 
    { 
     return x.SequenceEqual(y); 
    } 

    public int GetHashCode(List<SupplierCategory> obj) 
    { 
     return obj.GetHashCode(); 
    } 

    #endregion 
} 

而且我使用下面的LINQ查詢:

CategoryPathComparer comparer = new CategoryPathComparer(); 
List<List<SupplierCategory>> categoryPaths = (from i in infoList 
                  select 
                  new List<SupplierCategory>() { 
                  new SupplierCategory() { Name = i[3] }, 
                  new SupplierCategory() { Name = i[4], Parent = i[3] }, 
                  new SupplierCategory() { Name = i[5], Parent = i[4] }}).Distinct(comparer).ToList(); 

但明顯沒有做什麼,我想要它做,如下面的代碼演示:

comp.Equals(categoryPaths[0], categoryPaths[1]); //returns True 

我是否以錯誤的方式使用它?爲什麼他們沒有按照我的意圖進行比較?

編輯: 爲了證明該比較器不工作,下面的返回true,因爲它應該:

List<SupplierCategory> list1 = new List<SupplierCategory>() { 
    new SupplierCategory() { Name = "Cat1" }, 
    new SupplierCategory() { Name = "Cat2", Parent = "Cat1" }, 
    new SupplierCategory() { Name = "Cat3", Parent = "Cat2" } 
}; 
List<SupplierCategory> list1 = new List<SupplierCategory>() { 
    new SupplierCategory() { Name = "Cat1" }, 
    new SupplierCategory() { Name = "Cat2", Parent = "Cat1" }, 
    new SupplierCategory() { Name = "Cat3", Parent = "Cat2" } 
}; 
CategoryPathComparer comp = new CategoryPathComparer(); 
Console.WriteLine(comp.Equals(list1, list2).ToString()); 
+0

Retag。沒有C#3.5(請參閱http://stackoverflow.com/questions/247621/what-are-the-correct-version-numbers-for-c) – Vaccano 2009-11-02 23:10:28

回答

10

您的問題是你沒有正確地實現IEqualityComparer

當你實現IEqualityComparer<T>,你必須實現這樣GetHashCode任何兩個相等的對象具有相同的哈希碼。

否則,您會看到不正確的行爲,正如您在此處看到的。

您應該執行的GetHashCode如下:(單位:this answer提供)

public int GetHashCode(List<SupplierCategory> obj) { 
    int hash = 17; 

    foreach(var value in obj) 
     hash = hash * 23 + obj.GetHashCode(); 

    return hash; 
} 

還需要在SupplierCategory覆蓋GetHashCode要一致。例如:

public override int GetHashCode() { 
    int hash = 17; 
    hash = hash * 23 + Name.GetHashCode(); 
    hash = hash * 23 + Parent.GetHashCode(); 
    return hash; 
} 

最後,雖然你不需要,你應該重寫EqualsSupplierCategory並使其調用Equals方法你IEquatable實現。

4

實際上,這個問題甚至包含在文檔中: http://msdn.microsoft.com/en-us/library/bb338049.aspx

+0

您的意思是在社區內容? – SLaks 2009-11-03 04:17:23

+0

不,我的意思是該文檔包含VB和C#上的代碼示例,顯示如何創建自己的比較器。它還顯示瞭如何重寫GetHashCode和Equals方法。 – 2009-11-03 18:08:07