重要问题不是“Queryable.OfType做了什么,而是“我在那里看到的代码是如何实现的?”
反思 Queryable.OfType,我看到(经过一些清理):
public static IQueryable<TResult> OfType<TResult>(this IQueryable source)
{
return (IQueryable<TResult>)source.Provider.CreateQuery(
Expression.Call(
null,
((MethodInfo)MethodBase.GetCurrentMethod()).MakeGenericMethod(
new Type[] { typeof(TResult) }) ,
new Expression[] { source.Expression }));
}
所以让我看看我是否明白这一点:
- 使用反射来获取对当前方法 (OfType) 的引用。
- 新建一个方法,就是一模一样的,就是用MakeGenericMethod把当前方法的类型参数改成,呃,一模一样的东西。
- 该新方法的参数不是 source,而是 source.Expression。这不是 IQueryable,但我们会将整个内容传递给 Expression.Call,所以没关系。
- 调用 Expression.Call,
null
作为方法(奇怪?)实例传递,并将克隆的方法作为其参数传递。 - 将该结果传递给 CreateQuery 并转换结果,这似乎是整个事情中最明智的部分。
现在,此方法的效果是返回一个表达式,该表达式告诉提供者忽略返回类型不等于 TResult 或其子类型之一的任何值。但我看不到上述步骤实际上是如何实现这一点的。它似乎正在创建一个表示返回 IQueryable<TResult> 的方法的表达式,并使该方法的主体简单地成为整个源表达式,而无需查看类型。是否只是期望 IQueryable 提供程序只会默默地不返回任何不属于所选类型的记录?
那么上面的步骤在某种程度上是不正确的,还是我只是没有看到它们如何导致在运行时观察到的行为?