来源:https://note.guoqianfan.com/2021/03/21/debug-linq-with-vs/
前言
Linq调试有3种方法,准确来说是2种,因为LinqPad算是复制代码段到外部了。。
- VS自带调试:lambda表达式打断点
- VS插件OzCode
- LinqPad
VS自带调试
在VS里,是可以对Linq调试的,不过一般打断点都会打在整个语句上,这时候我们要换个打法,把断点打在lambda表达式上。
注意和前提
-
Linq是Linq to object
-
对于Linq to object,只有集合对象是
IEnumerable
时,才能命中到Linq里的lambda表达式,IQueryable
是不行的。如果是IQueryable
,此时就算在lambda表达式里打上断点,在代码执行时,断点会向上转移到整个语句上。如果是
IQueryable
,在lambda表达式里打上断点和设置操作,操作会输出错误:order name: id=error CS0103: 当前上下文中不存在名称“p”, name=error CS0103: 当前上下文中不存在名称“p”
。 -
对于Linq to object,当集合对象是
IEnumerable
时,是延迟执行的。只有结果被用到时,才会进行迭代。所以如果在实际执行前,集合数据发生改变会导致结果集和预期不符。 -
对于Linq to object,当集合对象是
IEnumerable
时,对单个对象进行迭代的方式是:先把单个对象走完所有的Linq方法后,直到最后或者执行到返回值不是IEnumerable
的Linq方法(该方法会被执行),才会迭代下一个对象。如果Linq方法的返回值不是
IEnumerable
,单个对象的迭代会到该方法(含)为止,会立即进行下一个对象的迭代。所有的对象迭代完毕后,会有一个临时的结果集(非IEnumerable
),然后把这个结果集重复前面的步骤,直至结束。OrderBy()
的返回值是IOrderedEnumerable
,所以运行了OrderBy()
后,单个对象的迭代就会结束,继续下一个对象的迭代。,然后把这个暂存结果集执行OrderBy()
后面的Linq方法。 -
在 4 的基础上,对于
IEnumerable
,如果有多个条件,我们可以写在同一个Where()
里,也可以拆开写在多个Where()
里,不会影响效率的,因为不会生成多个暂存结果集。 -
IEnumerable.AsQueryable().AsEnumerable()
是没问题的,遵循IEnumerable
的正常流程:断点不会转移,仍然是延迟执行。