如果您尝试在Visual Basic .NET中编译下面的查询,它将失败.
From x In {1,2} Select x.ToString()
编译器给出的错误是:
Range variable name cannot match the name of a member of the ‘Object’ class.
虽然等效的C#查询没有错,
from x in new[]{1,2} select x.ToString()
这不会发生在使用格式的ToString重载(它是Int32的成员,而不是Object).它与Object的其他成员确实发生,只要它们不带参数:GetType和GetHashCode失败;与等于(对象)它编译.
为什么这个限制到位,我可以使用什么替代品?
这是我的理解.请考虑以下代码:
Dim q = From x In {"Bob","Larry"} Select x.Length Select Length * 2
在上面的查询中,长度变量的名称由VB编译器根据表达式x.Length自动“猜测”.现在,你没有要求这个是真的这只是一个功能,无论你喜欢还是不喜欢它.但现在考虑一下:
Dim q = From x In {"Bob","Larry"} Select (x.Length) Select Length * 2
以上不编译,因为第一个Select子句中的表达式不像第一种情况那样简单(相信或不相信);括号使复杂的事情足够使编译器不选择名称长度;相反,它会生成一个不能从代码使用的名称.
所以基本上ToString()发生的情况是,这个表达式足够简单,使编译器可以用来生成一个变量名,如果查询扩展到使用这个变量,可以使用这个变量,例如:
Dim q = From x In { 1,2 } Select x.ToString() Select ToString.Length
但是,ToString不是变量的合法名称,因为它是System.Object的成员(为什么这是LINQ查询中的变量,而不是标准的局部变量,我不能说).