c# – 如何使用反射调用IDBSet.FirstOrDefault(谓词)?

给定一个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到对象的,看起来你正试图调用你的数据库.

点赞