给定一个IDbSet,其中Person包含“Id”属性,如何执行以下命令:
var p = PersonDbSet.FirstOrDefault(i=>i.Id = 3);
我可以构建谓词,并获得对FirstOrDefault扩展方法的引用,但我似乎无法将它们放在一起:
首先是谓词
ParameterExpression parameter = Expression.Parameter(entityType, "Id");
MemberExpression property = Expression.Property(parameter, 3);
ConstantExpression rightSide = Expression.Constant(refId);
BinaryExpression operation = Expression.Equal(property, rightSide);
Type delegateType = typeof (Func<,>).MakeGenericType(entityType, typeof (bool));
LambdaExpression predicate = Expression.Lambda(delegateType, operation, parameter);
现在引用扩展方法:
var method = typeof (System.Linq.Queryable).GetMethods(BindingFlags.Static | BindingFlags.Public)
.FirstOrDefault(m => m.Name == "FirstOrDefault" && m.GetParameters().Count() == 2);
MethodInfo genericMethod = method.MakeGenericMethod(new[] { entityType });
最后尝试执行该方法:
object retVal = genericMethod.Invoke(null, new object[] {dbSet, predicate});
使用以下消息抛出ArgumentException:
“类型’System.Reflection.RuntimePropertyInfo’的对象无法转换为’System.Linq.IQuerable`1 [Person]’类型.”
有什么想法吗?
最佳答案 您必须制作该方法的通用版本:
MethodInfo genericMethod = method.MakeGenericMethod(entityType);
object retVal = genericMethod.Invoke(dbSet, new object[] {expr});
顺便说一句.你不应该尝试从System.Linq.Queryable获取方法吗? System.Linq.Enumerable都是关于linq到对象的,看起来你正试图调用你的数据库.