在.NET中有两种类型的数组:一维的、基于零的(第一个索引是0)数组(
int[]
例如,由IL语言直接支持的
Expression
类)(作为旁注,它们被称为sz数组)和其他数组(多维数组,例如
int[5,3]
以及第一个索引与现有索引不同的数组,以与支持它们的某些语言(例如旧的vb)兼容,这些语言在IL语言中没有直接支持,但由使用对
Array
类。它们基于两个不同的类,这两个类都是
数组
. 正如我所写,低级的IL指令只适用于一维的、基于零的指令。安
数组
对象可以两者都是,因此没有来自的支持
Expression.ArrayLength
.
你可以很容易地看到这个
SharpLab
:
public static int A1<T>(T[] array) {
return array.Length;
}
public static int A2(Array array) {
return array.Length;
}
.method public hidebysig static
int32 A1<T> (
!!T[] 'array'
) cil managed
{
// Method begins at RVA 0x2050
// Code size 4 (0x4)
.maxstack 8
IL_0000: ldarg.0
IL_0001: ldlen
IL_0002: conv.i4
IL_0003: ret
} // end of method C::A1
.method public hidebysig static
int32 A2 (
class [mscorlib]System.Array 'array'
) cil managed
{
// Method begins at RVA 0x2055
// Code size 7 (0x7)
.maxstack 8
IL_0000: ldarg.0
IL_0001: callvirt instance int32 [mscorlib]System.Array::get_Length()
IL_0006: ret
} // end of method C::A2
只是出于好奇,使用了一个未记录的功能(我们在这里玩火!)…
ParameterExpression par = Expression.Parameter(typeof(Array), "array");
var conv = Expression.New(typeof(Conv));
var init = Expression.MemberInit(conv, Expression.Bind(typeof(Conv).GetField(nameof(Conv.Array)), par));
var array = Expression.Field(init, nameof(Conv.Array2));
var length = Expression.ArrayLength(array);
var lambda = Expression.Lambda<Func<Array, int>>(length, par);
var compiled = lambda.Compile();
在哪里?
Conv
是:
[StructLayout(LayoutKind.Explicit)]
public struct Conv
{
[FieldOffset(0)]
public Array Array;
[FieldOffset(0)]
public byte[] Array2;
}
使用方法如下:
int length = compiled(new int[100]);
这里的“诀窍”是通过
FieldOffest
我们可以强制转换具有相同内存布局的不兼容类型。所有sz数组共享相同的头格式(其中
Length
是包含的),所以我们“转换”我们的
数组
对
byte[]
数组(请注意,我们可能已经将其转换为任何其他类型,但在
byte
:它是最小的可用类型,因此我们确信不能以任何方式“走出”阵列)。
表达式树类似于:
static int GetLength(Array array)
{
return new Conv { Array = array }.Array2.Length;
}