2012-02-01 7 views
5

Utilizzando NUnit 2.5.9, il seguente test fallisce inaspettatamente:Comportamento errato di NUnit Assert.AreNotEqual quando si ha a che fare con IEnumerable <T>?

[TestFixture] 
public class FooTest 
{ 
    [Test] 
    public void Inequality() 
    { 
     var first = new Foo(new[] { 1 }, 2); 
     var second = new Foo(new[] { 1 }, 3); 

     Assert.AreNotEqual(first, second); 
    } 
} 

public struct Foo : IEnumerable<int>, IEquatable<Foo> 
{ 
    private readonly IEnumerable<int> values; 

    public int Bar { get; private set; } 

    public Foo(IEnumerable<int> values, int bar) 
     : this() 
    { 
     this.values = values; 
     Bar = bar; 
    } 

    public IEnumerator<int> GetEnumerator() 
    { 
     return values.GetEnumerator(); 
    } 

    IEnumerator IEnumerable.GetEnumerator() 
    { 
     return GetEnumerator(); 
    } 

    public bool Equals(Foo other) 
    { 
     return other.values.SequenceEqual(values) && other.Bar == Bar; 
    } 

    public override bool Equals(object obj) 
    { 
     if (ReferenceEquals(null, obj)) 
     { 
      return false; 
     } 
     if (obj.GetType() != typeof(Foo)) 
     { 
      return false; 
     } 
     return Equals((Foo)obj); 
    } 

    public override int GetHashCode() 
    { 
     unchecked 
     { 
      return ((values != null ? values.GetHashCode() : 0) * 397)^Bar; 
     } 
    } 

    public static bool operator ==(Foo left, Foo right) 
    { 
     return left.Equals(right); 
    } 

    public static bool operator !=(Foo left, Foo right) 
    { 
     return !left.Equals(right); 
    } 
} 

Scavando nel codice NUnit, si scopre che quando NUnit incontra due oggetti che implementano IEnumerable, solo confronta le due collezioni e ignora qualsiasi altra proprietà sugli oggetti.

Questo mi sembra sbagliato: il fatto che un oggetto implementa una particolare interfaccia non lo limita a svolgere solo quel ruolo. O l'interfaccia IEnumerable in .NET è un caso speciale? O ho appena frainteso NUnit?

+0

+1 Domanda interessante – sll

risposta

3

Sembra che questo sia un bug in NUnit, per quanto ho capito sarebbe stato corretto nella versione 3.0. Qui di seguito una discussione con un possibile lavoro intorno attuazione IComparer<T>:

+0

Grazie per le informazioni. Lavorerò intorno ad esso usando 'Assert.IsTrue (first! = Second)' per il momento – Akash

+0

@Akash: ma questo non controllerebbe anche proeprties, considera anche il controllo 'first.Bar! = Second.Bar' – sll

+1

@ sll: Perché no? La classe dell'OP ha un'implementazione '! =' Che chiama 'Equals', che poi controlla la proprietà' Bar' e gli elementi della collezione. – LukeH