我在ASP.NET MVC中创建了一个动态搜索屏幕。我通过反射从实体中检索了字段名称,这样我就可以允许用户选择要搜索的字段而不是显示视图中的所有字段。
当搜索结果回发到控制器时,我收到一个包含FieldName和Value的FormCollection。我不知道搜索了多少字段,FormCollection只包含用户选择的字段。
我希望能够现在获取该字段名称,并在我查询数据库时将其应用于我的LINQ语句,例如:
public List<People> SearchPeople(Dictionary<string, string> fieldValueDictionary)
{
List<People> searchResults = new List<People>();
foreach (string key in fieldValueDictionary.Keys)
{
searchResults.Add(entities.People.Where(p => p.<use the key string as the fieldName> == fieldValueDictionary[key]));
}
return searchResults;
}
我在“使用键字符串作为fieldName”的地方就像p => p.FirstName == fieldValueDictionary [key]其中key =“FirstName”。我尝试过但未能使用Lambda表树,并且在动态LINQ方面取得了一些成功。唯一的另一种选择是做类似的事情:
public List<People> SearchPeople(Dictionary<string, string> fieldValueDictionary)
{
IQueryable<People> results = entities.People;
foreach (string key in fieldValueDictionary.Keys)
{
switch (k)
{
case "FirstName": results = results.Where(entities.People.Where(p => p.FirstName == k);
case "LastName": results = results.Where(entities.People.Where(p => p.LastName == k);
// Repeat for all 26 fields in table
}
}
return results.ToList<People>();
}
更新 :我通过以下帖子研究了Lambda Expression Trees:
动态创建lambdas表达式+ linq + OrderByDescending
LINQ:将lambda表达式作为参数传递,由方法执行并返回
我已经得到了一个lambda输出以下内容:“p => p.FirstName”,但我不能让它在哪里工作。有什么建议么?我的代码如下:
MemberInfo member = typeof(People).GetProperty("FirstName");
ParameterExpression cParam = Expression.Parameter(typeof(People), "p");
Expression body = Expression.MakeMemberAccess(cParam, member);
var lambda = Expression.Lambda(body, cParam);
经过更多的反复试验和搜索,我意外地找到了另一篇涉及同一问题的SO帖子:
InvalidOperationException:类型'System.Linq.Queryable'上没有方法'Where'与提供的参数兼容
这是我修改后的代码:
IQueryable query = entities.People;
Type[] exprArgTypes = { query.ElementType };
string propToWhere = "FirstName";
ParameterExpression p = Expression.Parameter(typeof(People), "p");
MemberExpression member = Expression.PropertyOrField(p, propToWhere);
LambdaExpression lambda = Expression.Lambda<Func<People, bool>>(Expression.Equal(member, Expression.Constant("Scott")), p);
MethodCallExpression methodCall = Expression.Call(typeof(Queryable), "Where", exprArgTypes, query.Expression, lambda);
IQueryable q = query.Provider.CreateQuery(methodCall);
通过一些希望非常简单的修改,我应该能够使用任何类型。
再次感谢您的答案Ani和John Bowen
您是否尝试过从PropertyInfo获取价值?
entities.People.Where(p => (p.GetType().GetProperty(key).GetValue(p, null) as string) == fieldValueDictionary[key])