在 linq to entities 中使用自定义方法

问题描述 投票:0回答:2

我的数据库中有一个

Person
表,其中包含
NationalId
字段。 有什么方法可以使用
NationalId
Ef code first
加载所有具有偶数
Linq to entities
的人,而无需将所有
Person
加载到内存中?

类似的东西:

public  bool IsEven(int number)
{
   return number % 2 == 0;
}

var context = new MyContext();
var personsWithEvenNationalId = context.Persons
                                       .Where(x=> IsEven(x.NationalId))
                                       .ToList();
c# entity-framework ef-code-first linq-to-entities
2个回答
4
投票

您必须在线检查

var personsWithEvenNationalId = context.Persons
                                       .Where(x=> x.NationalId%2 == 0)
                                       .ToList();

Linq to Entities 基本上不知道如何将自定义方法转换为 SQL。如果您确实需要使用自定义方法,则必须将 Persons 设置为可枚举的,然后使用您的自定义方法,即

var personsWithEvenNationalId = context.Persons
                                       .AsEnumerable()
                                       .Where(x=> IsEven(x.NationalId))
                                       .ToList();

但这并不理想,因为它会加载所有 Persons,然后过滤 IsEven

编辑: 考虑一下,如果您不想每次都内联编写,您也可以为

IQueryable<Person>
创建一个扩展方法。像这样的东西,你在那里建立一个
Expression

    public static IQueryable<Person> WhereEven(this IQueryable<Person> source, Expression<Func<Person, int>> property)
    {
        var expression = Expression.Equal(
            Expression.Modulo(
                property.Body,
                Expression.Constant(2)),
            Expression.Constant(0));

        var methodCallExpression = Expression.Call(typeof (Queryable),
            "where",
            new Type[] {source.ElementType},
            source.Expression,
            Expression.Lambda<Func<Person, bool>>(expression, property.Parameters));

        return source.Provider.CreateQuery<Person>(methodCallExpression);
    }

使用它:

context.Persons.WhereEven(x => x.NationalId).ToList();

1
投票

您需要一个函数(或属性或字段)来提供

Expression
来执行您想要的投影,而不是一个函数来执行您想要的操作:

public static Expression<Func<int, bool>> IsEven()
{
    return number => number % 2 == 0;
}

你现在可以写:

using(var context = new MyContext())
{
    var personsWithEvenNationalId = context.Persons
        .Select(x=> x.NationalId)
        .Where(IsEven())
        .ToList();
}
© www.soinside.com 2019 - 2024. All rights reserved.