走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq

系列文章导航:

走进Linq--Linq横空出世篇

走进Linq-辉煌的背后

走进Linq-Linq大观园

不能不说的C#特性-对象集合初始化器

不能不说的C#特性-匿名类型与隐式类型局部变量

不能不说的C#特性-扩展方法

不能不说的C#特性-匿名方法和Lambda表达式

不能不说的C#特性-迭代器(上)及一些研究过程中的副产品

不能不说的C#特性-迭代器(下),yield以及流的延迟计算

走进Linq-Linq to Objects(上)基础篇

走进Linq-Linq to Objects(下)实例篇

走进Linq-Linq to SQL感性认识篇

走进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-How do I(4)拾遗补零篇第一节

走进Linq-Linq to SQL源代码赏析 Table的获取过程

走进Linq-Linq to SQL源代码赏析之Provider的初始化

走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq


LINQ,语言集成查询,就是把一些查询操作集成到语言中(貌似是废话),比如查询关系数据库,而且提供一种一致的操作方式,不管最终的数据存储在哪里?内存中,远程数据库还是一Xml格式文件存储,不仅仅如此,你还可以用你丰富的想象力扩充自己的查询。Linq to SQL无疑把Linq的能量发挥的淋漓尽致,我们就以Linq to SQL的体系结构来学习一下Linq的整体框架。

在上两章里面我们通过源代码探讨了关于DataContext的初始化和Table<TEntity>对象的获取,以及Provider的初始化。今天我们来看看Linq to SQL执行的大至流程

假如我们写下这样的代码:

DataContext dbCtx = new DataContext("server=localhost;database=cnblogs;user id=sa;pwd=sa");
Table
<Post> posts = dbCtx.GetTable<Post>();
foreach (Post p in posts)
{
   Console.WriteLine(p.Title);
}

系列文章导航:

走进Linq--Linq横空出世篇

走进Linq-辉煌的背后

走进Linq-Linq大观园

不能不说的C#特性-对象集合初始化器

不能不说的C#特性-匿名类型与隐式类型局部变量

不能不说的C#特性-扩展方法

不能不说的C#特性-匿名方法和Lambda表达式

不能不说的C#特性-迭代器(上)及一些研究过程中的副产品

不能不说的C#特性-迭代器(下),yield以及流的延迟计算

走进Linq-Linq to Objects(上)基础篇

走进Linq-Linq to Objects(下)实例篇

走进Linq-Linq to SQL感性认识篇

走进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-How do I(4)拾遗补零篇第一节

走进Linq-Linq to SQL源代码赏析 Table的获取过程

走进Linq-Linq to SQL源代码赏析之Provider的初始化

走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq


 

既然如此那我们就得看看Table<TEntity>GetEnumerator方法了:

public IEnumerator<TEntity> GetEnumerator()
{
   
return ((IEnumerable<TEntity>)this.context.Provider.Execute(Expression.Constant(this)).ReturnValue).GetEnumerator();
}

系列文章导航:

走进Linq--Linq横空出世篇

走进Linq-辉煌的背后

走进Linq-Linq大观园

不能不说的C#特性-对象集合初始化器

不能不说的C#特性-匿名类型与隐式类型局部变量

不能不说的C#特性-扩展方法

不能不说的C#特性-匿名方法和Lambda表达式

不能不说的C#特性-迭代器(上)及一些研究过程中的副产品

不能不说的C#特性-迭代器(下),yield以及流的延迟计算

走进Linq-Linq to Objects(上)基础篇

走进Linq-Linq to Objects(下)实例篇

走进Linq-Linq to SQL感性认识篇

走进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-How do I(4)拾遗补零篇第一节

走进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,它还有一个属性是BodyBody也是一个Expression类型,从这里我们看到Expression是这样递归下去的。

通过上面的介绍,实际上Queryable中的那些扩展方法所接受的Lambda表达式最后被编译为树数据结构。这些树数据结构携带有查询表达式的语法,但是最终它们要查询什么样的数据,是数据库?还是XML或是Web Service就要靠IQueryProvider来解析了,从这里我们大概可以看到这样一个结构:

看到这个图,那我们有几种扩展Linq的方式呢?

第一种:通过给IEnumerableIQueryable添加扩展方法,这个就是利用C#语言的特性来达到的。我将这种扩展称之为横向的扩展。

第二种:自己实现IQueryProvider,这种扩展就是纵向的扩展了,提供自己的Provider,然后这个Provider解析表达式树生成最终具体的查询操作。

小结

本文简单的展示一下Linq的体系结构,了解一下Linq的扩展点在哪里,从这里我们也能看到,要做一个好扩展的系统,一个很重要的任务就是提炼接口,接口的粒度,接口的职责等等都是核心关注点。使用接口将几个类隔离,变化点也就封装起来了。

it知识库走进Linq-Linq to SQL源代码赏析,通过Linq to SQL看Linq,转载需保留来源!

郑重声明:本文版权归原作者所有,转载文章仅为传播更多信息之目的,如作者信息标记有误,请第一时间联系我们修改或删除,多谢。