走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq
[2] 走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq
[3] 走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq
系列文章导航:
不能不说的C#特性-迭代器(下),yield以及流的延迟计算
走进Linq-Linq to SQL How do I(1)
走进Linq-Linq to SQL How do I(2)
走进Linq-Linq to SQL How do I(3)
走进Linq-Linq to SQL源代码赏析 Table
走进Linq-Linq to SQL源代码赏析之Provider的初始化
走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq
赋值符号右边的表达式是一样的,在第一个式子中是将Lambda表达式赋值给委托,第二个式子中将Lambda赋值给Expression,这个东西叫做表达式树。
在第一个式子编译的时候后面的表达式实际上会被转换为一个匿名方法,IsTrue也就是一个“方法的指针”,和我们已经熟识的委托没有什么区别。而第二个式子在执行的时候右边的表达式会被编译为一个树的数据结构(C#编译器实际上为我们做了词法分析、语法分析了,在编译原理里前两个阶段就是词法分析和语法分析,词法分析首先遍历传入的语言字符串,在我们这里就是x=>x==5,词法分析器读取每个字符,识别出标识符,常量,关键字,运算符,词法分析器的产出是Token(符号);然后语法分析器根据该语言的语法范式将Token组织成一个树形结构,用这个树形的结构来表示该语言的代码文件,在我们这里x=>x==5就是Lambda表达式这门“语言”的语句了,Expression就是那个树)。Expression是一个递归形式的定义,它有两个属性:Parameters,这个属性就是Lambda表达式的参数,在上面的代码中就是:x,它还有一个属性是Body,Body也是一个Expression类型,从这里我们看到Expression是这样递归下去的。
通过上面的介绍,实际上Queryable中的那些扩展方法所接受的Lambda表达式最后被编译为树数据结构。这些树数据结构携带有查询表达式的语法,但是最终它们要查询什么样的数据,是数据库?还是XML或是Web Service就要靠IQueryProvider来解析了,从这里我们大概可以看到这样一个结构:
看到这个图,那我们有几种扩展Linq的方式呢?
第一种:通过给IEnumerable或IQueryable添加扩展方法,这个就是利用C#语言的特性来达到的。我将这种扩展称之为横向的扩展。
第二种:自己实现IQueryProvider,这种扩展就是纵向的扩展了,提供自己的Provider,然后这个Provider解析表达式树生成最终具体的查询操作。
小结
本文简单的展示一下Linq的体系结构,了解一下Linq的扩展点在哪里,从这里我们也能看到,要做一个好扩展的系统,一个很重要的任务就是提炼接口,接口的粒度,接口的职责等等都是核心关注点。使用接口将几个类隔离,变化点也就封装起来了。