2

我有 1 个使用了这么多的 LINQ。我尝试创建返回此 LINQ 的方法,例如:

    public static System.Linq.Expressions.Expression<Func<MyEntity, bool>> GetFilteredEntity() {
        return x => true/*Some condition*/;
    }

    public static Func<MyEntity, bool> GetFilteredEntity() {
        return x => true/*Some condition*/;
    }

并像这样使用

    db.MyEntities.Where(GetFilteredEntity());

成功了,但是!我需要像这样使用它

    db.ParentEntities.Where(entity => entity.MyEntities.Where(GetFilteredEntity()));

这段代码也编译了,但每次我使用它时,我都会收到错误:

System.InvalidOperationException: There is already an open DataReader associated with this Command which must be closed first.

,甚至:

db.ParentEntities.Where(entity => entity.MyEntities.Where(GetFilteredEntity())).ToList();

也抛出这个异常。

但,

db.ParentEntities.Where(entity => entity.MyEntities.Where(x => true/*Some condition*/))

仍然可以正常工作!那么为什么会发生这种情况,并且有办法解决这个问题呢?

最终工作代码

public static Expression<Func<MyEntity, bool>> GetFilteredEntity() { return x => true/*Some condition*/; }

var expression = GetFilteredEntity();

db.ParentEntities.Where(entity => entity.MyEntities.AsQueryable().Where(expression ));

还要.AsQueryable()感谢在 Linq to Entities 中将 func 作为参数传递和“Internal .NET Framework Data Provider error 1025”错误

4

2 回答 2

4

在您的第一个示例中,该函数在发送到查询提供程序之前被调用并转换为表达式。在接下来的两个示例中,函数调用嵌入在发送到查询提供程序的表达式中,并且该查询提供程序不知道如何处理该函数调用,所以它只是抛出一个异常。当您将实际表达式嵌入到另一个表达式中时,不会调用函数来混淆查询提供程序。

至于解决方案,只需将函数调用拉出到变量中即可。查询提供程序足够聪明,可以看到您使用了封闭变量,并将提取其值。对于函数调用,它只是不确定是否应该评估它或尝试将其转换为应该在数据库端完成的事情。对于查询提供者和使用它的人来说,尝试两者都做会非常令人困惑且难以使用。为了简化问题,在发送查询之前永远不会执行带有表达式的函数调用。至于封闭变量,没有其他方法可以处理它,因此没有任何其他行为可以混淆它。

var expression = GetFilteredEntity();
db.ParentEntities.Where(entity => entity.MyEntities.Where(expression ));
于 2013-10-07T16:01:02.977 回答
-1

看起来 LazyLoading 可能是罪魁祸首,您是否尝试过在参数上弹出 ToList() ?

db.ParentEntities.Where(entity => entity.MyEntities.Where(GetFilteredEntity()).ToList());
于 2013-10-07T16:00:29.347 回答