我有一个非常类似的查询:
var result = context.EntityRole
.Where(er => er.EntityType.EntityTypeId == entityTypeIdParameter
&& er.Entity.SomeItems.Any(item => item.ItemId == itemIdParameter))
.ToList()
.Distinct(customItemComparer)
.OrderBy(er => er.Id)
.ThenByDescending(er => er.IsApproved)
.ToList();
customItemComparer类似于以下内容:
public class CustomItemComparer : IEqualityComparer<EntityRole>
{
public bool Equals(EntityRole x, EntityRole y)
{
if (ReferenceEquals(x, y)) return true;
if (ReferenceEquals(x, null) || ReferenceEquals(y, null)) return false;
return x.Property1 == y.Property1
&& x.Property2 == y.Property2
&& x.Property3 == y.Property3;
}
public int GetHashCode(EntityRole obj)
{
if (ReferenceEquals(obj, null)) return 0;
var hasProperty1 = obj.Property1.GetHashCode();
var hasProperty2 = obj.Property2.GetHashCode();
var hasProperty3 = obj.Property3.GetHashCode();
return hasProperyt1 ^ hasProperty2 ^ hasProperty3;
}
}
我的问题是,当我运行应用程序时,我得到了预期的结果,显然所有不同的情况都运行良好,但是当我尝试单元测试时,查询始终返回单个对象,即使列表包含多个具有不同属性1、2和3的对象。
我的单元测试大致如下,我们使用MOQ,在此简化其他属性:
var roles = new List<EntityRole>
{
new EntityRole
{
Property1 = true,
Property2 = 5,
Property3 = "something"
},
new EntityRole
{
Property1 = true,
Property2 = 9,
Property3 = "something"
},
new EntityRole
{
Property1 = false,
Property2 = 5,
Property3 = "something"
},
new EntityRole
{
Property1 = true,
Property2 = 5,
Property3 = "something else"
}
}
contextMock.Setup(c => c.EntityRole).Returns(roles.AsQueryable);
var sut = new SubjectUnderTest();
sut.MethodWhereQueryIsExecuted();
//some code to verify results
就像我所说的那样,即使在列表中没有两个相同的对象,查询始终返回第一个对象。
此外,如果我在CustomItemComparer中设置断点,应用程序运行时会停止执行,但调试测试时却永远不会停止。
因此,确切的问题是:为什么Distinct在运行应用程序时完美地工作,而在运行单元测试时却不起作用?
contextMock.Setup(c => c.EntityRole).Returns(roles.AsQueryable);
没有任何意义。Setup
方法应该模拟某个对象的方法,而Returns
应该返回该方法的一些虚假结果。我没有看到你的 contextMock 调用了带有该查询的 db 查询或方法。 - rock_walkerToList
,而是想要移除之后的所有内容,以检查Where
或Distinct
是否过滤了其他项... - Christoph Fink