参数化Linq表达式帮助

Lui*_*lar 5 c# linq linq-to-entities expression-trees

我想用这样的签名做一个方法:

Expression<Func<TSource, bool>> CreatePropertyFilter<TSource>(Expression<Func<TSource, string>> selector, string value, TextMatchMode matchMode);
Run Code Online (Sandbox Code Playgroud)

基本上,它需要一个属性选择器(例如:p = p.Name),一个字符串值,并且其可以是枚举值StartsWith,EndsWith,Contains,Exact; 用于文本匹配选项.

如何以LINQ2Entities可以理解的方式实现该方法?我已经使用嵌套的调用表达式实现了这个方法,如下所示:

Expression<Func<string, bool>> comparerExpression;

switch (matchMode)
{
    case TextMatchMode.StartsWith:
       comparerExpression = p => p.StartsWith(value);
       break;
    case TextMatchMode.EndsWith:
       comparerExpression = p => p.EndsWith(value);
       break;
    case TextMatchMode.Contains:
       comparerExpression = p => p.Contains(value);
       break;
    default:
       comparerExpression = p => p.Equals(value);
       break;
}

var equalityComparerParameter = Expression.Parameter(typeof(IncomingMail), null);
var equalityComparerExpression = Expression.Invoke(comparerExpression, Expression.Invoke(selector, equalityComparerParameter));
var equalityComparerPredicate = Expression.Lambda<Func<IncomingMail, bool>>(equalityComparerExpression, equalityComparerParameter);
Run Code Online (Sandbox Code Playgroud)

问题是Linq2Entities不支持调用表达式.

有什么建议吗?

谢谢!

Ani*_*Ani 5

基本上,给定一个选择器:

input => input.Member
Run Code Online (Sandbox Code Playgroud)

您当前正在构建谓词表达式,如:

input => selector(input).Method(value)
Run Code Online (Sandbox Code Playgroud)

相反,通过使用它的主体(a MemberExpression)"展开"选择器表达式来构造类似于:

input => input.Member.Method(value) 
Run Code Online (Sandbox Code Playgroud)

这看起来像:

private static Expression<Func<TSource, bool>> CreatePropertyFilter<TSource>
    (Expression<Func<TSource, string>> selector, 
     string value, 
     TextMatchMode matchMode)
{
    // Argument-checking here.    

    var body = selector.Body as MemberExpression;

    if (body == null)
        throw new ArgumentException("Not a MemberExpression.");    

    // string.StartsWith / EndsWith etc. depending on the matchMode.
    var method = typeof(string)
                 .GetMethod(GetMethodName(matchMode), new[] { typeof(string) });

    // input.Member.method(value)
    var compEx = Expression.Call(body, method, Expression.Constant(value));

    // We can reuse the parameter of the source.
    return Expression.Lambda<Func<TSource, bool>>(compEx, selector.Parameters);
}
Run Code Online (Sandbox Code Playgroud)

翻译方法是:

// I really don't like this enum.
// Why not explicitly include Equals as a member?
private static string GetMethodName(TextMatchMode mode)
{
    switch (mode)
    {
        case TextMatchMode.StartsWith:
            return "StartsWith";

        case TextMatchMode.EndsWith:
                return "EndsWith";

        case TextMatchMode.Contains:
            return "Contains";

        default:
            return "Equals";
    }    
}
Run Code Online (Sandbox Code Playgroud)