2012-02-01 42 views
5

使用NUnit 2.5.9,下面的测试意外失败:处理IEnumerable时NUnit Assert.AreNotEqual的行为不正确<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); 
    } 
} 

挖掘到NUnit的代码,事实证明,当NUnit的遇到一个实现IEnumerable两个对象,它只是比较了两个集合,而忽略其他任何物体上的属性。

这对我来说是错误的:对象实现特定接口的事实并不限制它只执行该角色。或者是.NET中的IEnumerable接口是一种特殊情况?或者我误解了NUnit?

+0

+1有趣的问题 – sll 2012-02-01 11:57:19

回答

3

看起来这是NUnit中的一个错误,据我了解,它将在3.0版本中修复。下面有一个可能的解决办法实现IComparer<T>一些讨论:

+0

感谢您的信息。我将通过'Assert.IsTrue(first!= second)'来解决它暂时的问题 – Akash 2012-02-01 13:22:29

+0

@Akash:但是这并不会检查proeprties,也可以考虑检查'first.Bar!= second.Bar' – sll 2012-02-01 13:41:23

+1

@sll:为什么不呢? OP的类有一个'!='实现,调用'Equals',然后检查'Bar'属性以及集合的元素。 – LukeH 2012-02-01 14:33:31