4

方法 Array.SetValue(Object value, int index) 允许涉及值/数组类型对的赋值,这通常允许使用通用索引器语法,并在您尝试组合通常不允许的类型时抛出异常。例如,考虑以下局部变量声明:

int[] twoints = new int[2] { 5, 6 };

以下四行不会引发任何运行时或编译时异常:

twoints[1] = (sbyte)7;
twoints.SetValue((sbyte)7, 1);
twoints[1] = (char)7;
twoints.SetValue((char)7, 1);

另一方面,这四行中的每一行都会在运行时或设计时抛出异常:

twoints[1] = 4.5;
twoints.SetValue(4.5, 1);
twoints[1] = 4L;
twoints.SetValue(4L, 1);

但是,当我将字节值分配给 char 数组时,我会得到奇怪的结果。索引器语法在编译时被拒绝,并且 SetValue 的 API 调用在运行时成功:

char[] twochars = new char[2] { 'A', 'B' };
twochars[1] = (byte)70;                 // Not OK, refused by the compiler
twochars.SetValue((byte)70, 1);         // OK, no exception at run-time

为什么允许此操作?

4

1 回答 1

2

这是因为Array.SetValue()遵循 CLR 规则,但赋值遵循 C# 类型安全规则。

在大多数情况下,小整数类型的局部变量和临时变量实际上native int在 CIL 中使用。“类型”是由 C# 编译器创建和强制执行的一种错觉。

在赋值中,您native int在 IL 操作数堆栈上有一个,C# 编译器基于类型System.Byte和 C# 规则执行类型检查,用于整数类型的隐式转换。CLR 非常乐意将值写入数组...错误消息是由 C# 编译器添加的。它可能应该是一个警告,但 C# 语言设计者选择指定它是一个错误。

Array.SetValue中,您有一个实际的装箱值 type System.Byte,以及由您调用的方法执行的转换(可能是通过委托给其他函数)。

于 2013-05-20T16:02:52.347 回答