我想为我的项目实现通用寻呼机和过滤器 View 模型,但我一直坚持使用反射获取 OrderBy 方法。这是我尝试过的方法,但对于 methodInfo
,一直得到 null
。似乎我将错误的 Type[]
参数传递给 GetMethod()
方法,但我无法正确处理。
protected virtual Expression<Func<T, IComparable>> GetOrderByExpression()
{
var type = typeof(T);
var property = type.GetProperty("DataSetName");
var parameter = Expression.Parameter(type, "x");
var propertyAccess = Expression.MakeMemberAccess(parameter, property);
var orderByExp = Expression.Lambda(propertyAccess, parameter);
var methodInfo = typeof(Enumerable).GetMethod("OrderBy", new Type[] { orderByExp.Body.Type });
var predicateBody = Expression.Call(propertyAccess, methodInfo, orderByExp);
var expression = Expression.Lambda<Func<T, IComparable>>(predicateBody, parameter);
return expression;
}
最佳答案
然后 Enumerable.OrderBy
扩展方法是一个静态方法,因此您必须使用 GetMethod
的重载,您可以在其中指定 BindingFlags
(BindingFlags.Static | BindingFlags.Public
而不是默认的 BindingFlags.Instance | BindingFlags.Public
)。
您还必须为该方法指定两个参数 - 从语法上看,它看起来只有一个参数,但因为它是一种扩展方法,所以您必须指定第二个参数。
这是你想通过反射得到的方法:
public static IOrderedEnumerable<TSource> OrderBy<TSource, TKey>(
this IEnumerable<TSource> source,
Func<TSource, TKey> keySelector
)
您需要两种参数类型:
var sourceType = typeof(IEnumerable<>).MakeGenericType(type);
var keySelectorType = orderByExp.Compile().GetType();
似乎只有执行搜索才能获得具有通用参数的方法的特定重载。幸运的是,Enumerable.OrderBy
的重载只有两个,而您想要的是带有两个参数的那个:
var genericMethodInfo = typeof(Enumerable)
.GetMethods(BindingFlags.Static | BindingFlags.Public)
.First(mi => mi.Name == "OrderBy" && mi.GetParameters().Length == 2);
var methodInfo = genericMethodInfo.MakeGenericMethod(sourceType, keySelectorType);
这将返回所需的 MethodInfo
,但是您必须修改剩余的两行代码,因为此方法是静态方法并且您必须在 Expression.Call 中指定一个空实例
。
关于c# - 使用反射获取 OrderBy 方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29386896/