我正在使用 XUnit 框架来测试我的 C# 代码。
在这个框架中是否有任何可用的断言方法来进行对象比较?我的目的是检查每个对象的公共和私有成员变量是否相等。
我尝试了这些替代方案,但很少奏效:
1) bool IsEqual = (Obj1 == Obj2)
2) Assert.Same(Obj1, Obj2) which I couldnt understand what happens internally
我有类似的问题,但幸运的是我已经在使用
using Newtonsoft.Json;
所以我只需要将它序列化为 json 对象,然后作为字符串进行比较。
var obj1Str = JsonConvert.SerializeObject(obj1);
var obj2Str = JsonConvert.SerializeObject(obj2);
Assert.Equal(obj1Str, obj2Str );
FluentAssertions库内部有一些非常强大的比较逻辑。
myObject.ShouldBeEquivalentTo(new { SomeProperty = "abc", SomeOtherProperty = 23 });
您甚至可以使用它来断言“myObject”的一部分。但是,它可能无法帮助您处理私有字段。
您需要有一个自定义比较器来实现这一点,当您比较对象时,它们会根据它们是否引用内存中的同一对象来检查。要覆盖此行为,您需要覆盖Equals
andGetHashCode
方法,然后您可以执行以下操作:
Assert.True(obj1.Equals(obj2));
这是一个 MSDN 页面 abt 重载 Equals 方法: http: //msdn.microsoft.com/en-us/library/ms173147 (v=vs.80).aspx
也可以对这个问题发表评论:IEquatable 和只是覆盖 Object.Equals() 有什么区别?
有 NuGet 包可以为您执行此操作。这是我个人使用的两个示例。
深度相等:
object1.ShouldDeepEqual(object2);
预期对象:
[Fact]
public void RetrievingACustomer_ShouldReturnTheExpectedCustomer()
{
// Arrange
var expectedCustomer = new Customer
{
FirstName = "Silence",
LastName = "Dogood",
Address = new Address
{
AddressLineOne = "The New-England Courant",
AddressLineTwo = "3 Queen Street",
City = "Boston",
State = "MA",
PostalCode = "02114"
}
}.ToExpectedObject();
// Act
var actualCustomer = new CustomerService().GetCustomerByName("Silence", "Dogood");
// Assert
expectedCustomer.ShouldEqual(actualCustomer);
}
我知道这是一个老问题,但自从我偶然发现它后,我想我会考虑一个可用的新解决方案(至少在.net Core 2.0 解决方案中的 xunit 2.3.1 中)。
我不确定它是什么时候引入的,但现在有一个重载形式,.Equal
它接受一个实例IEqualityComparer<T>
作为第三个参数。您可以在单元测试中创建自定义比较器,而不会污染您的代码。
可以像这样调用以下代码:Assert.Equal(expectedParameters, parameters, new CustomComparer<ParameterValue>());
XUnit 本身似乎会在遇到故障时立即停止处理测试,因此EqualException
从我们的比较器中抛出一个新的似乎与 XUnit 开箱即用的工作方式是一致的。
public class CustomComparer<T> : IEqualityComparer<T>
{
public bool Equals(T expected, T actual)
{
var props = typeof(T).GetProperties(BindingFlags.DeclaredOnly | BindingFlags.Public | BindingFlags.Instance);
foreach (var prop in props)
{
var expectedValue = prop.GetValue(expected, null);
var actualValue = prop.GetValue(actual, null);
if (!expectedValue.Equals(actualValue))
{
throw new EqualException($"A value of \"{expectedValue}\" for property \"{prop.Name}\"",
$"A value of \"{actualValue}\" for property \"{prop.Name}\"");
}
}
return true;
}
public int GetHashCode(T parameterValue)
{
return Tuple.Create(parameterValue).GetHashCode();
}
}
编辑:我发现比较实际值和预期值!=
对某些类型无效(我确信有一个更好的解释涉及引用类型和值类型之间的差异,但今天不是这样)。我更新了代码以使用该.Equals
方法来比较两个值,这似乎效果更好。
这是 IEqualityComparer 接口的另一个实现,具有真正的GenericComparer
并且为了断言 2 个相同类型的列表是相等的,无论列表容器如何,您都可以添加以下扩展:
public static bool IsEqualTo<T>(this IEnumerable<T> list, IEnumerable<T> expected) where T : class
{
if (!list.HasSameLengthThan<T>(expected))
return false;
Assert.Equal(expected, list, new GenericComparer<T>());
return true;
}