2

我有一个简单的 linq-to-sql 语句,它将 astring[]中的值与表中的值匹配并返回匹配的表值。

_table是类型System.Data.Linq.Table<TEntity>

public IEnumerable<ZipCode> match(IEnumerable<string> values) {
    HashSet<string> valueSet = new HashSet<string>(values);
    return _table.Where(x => valueSet.Contains(x.zipcode);
}

我怎样才能重构它,以便我可以将要使用的字段传递到Contains()?

我失败的尝试:

public IEnumerable<ZipCode> match<T>(IEnumerable<T> values, 
        Expression<Func<ZipCode, T>> field) {
    HashSet<T> valueSet = new HashSet<T>(values);
    return _table.Where(x => valueSet.Contains(field(x)));
}

这会产生错误:

方法“System.Object DynamicInvoke(System.Object[])”不支持 SQL 转换。

我的想法是,我可以使用表达式树为字段提供字段,Contains()但对如何或是否是使用表达式的正确方法感到困惑。可能有一个更好的解决方案。

4

2 回答 2

2

我能够使用以下内容(在 .NET 4.5 中,但我相信它可以在早期版本中使用)

static IEnumerable<TTable> Match<TTable, TField>(IQueryable<TTable> table, IEnumerable<TField> values, Expression<Func<TTable, TField>> fieldSelector)
{
    var valuesConstant = Expression.Constant(values.Distinct());

    var callContains = Expression.Call(typeof(Enumerable), "Contains", new[] { typeof(TField) }, new Expression[] { valuesConstant, fieldSelector.Body });

    var whereExpression = Expression.Lambda<Func<TTable, bool>>(callContains, (fieldSelector.Body as MemberExpression).Expression as ParameterExpression);

    return table.Where(whereExpression);
}

您应该能够删除我使用的表参数并使用您拥有的成员变量。

于 2013-02-06T00:12:28.687 回答
0

而不是Expression<Func<ZipCode, T>> field使用Func<ZipCode, T> field

public IEnumerable<ZipCode> match<T>(IEnumerable<T> values, Func<ZipCode, T> field)
{
    HashSet<T> valueSet = new HashSet<T>(values);
    return _table.Where(x => valueSet.Contains(field(x)));
}
于 2013-02-05T15:53:28.027 回答