0

我从一位前雇员那里继承了一些代码,该代码使用编译的 Linq 表达式和MongoRepository 库(位于 MongoDB C# 驱动程序之上)查询 mongo DB。

这些需要很长时间才能运行 - 通常大约 6 分钟(!)并导致使用它们的控制器方法出现问题。

所以我简化了代码并删除了.Compile()对 lambda 表达式的调用,这似乎解决了这个问题(现在运行需要 <10 秒)。

我的问题是:为什么在查询 mongo 时编译这个表达式会导致问题?

这是原始代码的要点(被黑了,所以断章取义):

public class BaseMongoRepository<T> : MongoRepository<T, Guid> where T : IEntity<Guid> {

    protected BaseMongoRepository(string connectionString) : base(connectionString) { }

    protected bool IsSatisfiedBy(T entity) {
        Expression<Func<T, bool>> func = x => x != null && x.ToString() == "foo"; // query was passed in, but you get the idea
        var predicate = func.Compile(); // THIS LINE??
        return predicate(entity);
    }

    public IEnumerable<T> Find() {
        return base.collection.AsQueryable().Where(IsSatisfiedBy);
    }
}

我将它简化为只使用常规谓词的东西Func

public IEnumerable<T> Find() {
    return base.collection.AsQueryable().Where(x => x != null && x.ToString() == "foo");
}

任何想法都非常感谢!

4

2 回答 2

1

提供者可以将表达式转换为真正的 sql 查询,但不能解释委托。
此代码中的predicate变量:

Expression<Func<T, bool>> func = x => x != null && x.ToString() == "foo";
var predicate = func.Compile();

本质上是一样的:

Func<T, bool> predicate = x => x != null && x.ToString() == "foo";

当您使用这样的委托时,数据库中的所有数据都将传输到内存中,然后应用谓词。

伪代码示例:

// Using the delegate:
var data = dbContext.Users.Where(usr => IsSatisfiedBy(usr)).ToList();
// This will result in the following steps:
var userList = ExecuteQuery("SELECT * FROM Users"); // all users are fetched.
var satisfied = userList.Where(usr => IsSatisfiedBy(usr))

// Using an expression:
var data = dbContext.Users.Where(usr => usr.Name == "foo");
// This will result in the following step:
var satisfied = ExecuteQuery("SELECT * FROM Users WHERE Name = 'foo'"); // Filtered before returned to caller.
于 2019-01-31T09:54:49.647 回答
-1

性能问题的原因是记录了相关对象的所有记录,然后过滤掉。首先,您需要从 mongo db 创建查询并注册。

查看源代码 https://github.com/fsefacan/MongoDbRepository

于 2019-01-25T22:45:02.633 回答