无法将ref enum转换为ref动态?

时间:2013-03-23 00:54:19

标签: c# c#-4.0 dynamic enums

给出函数声明

dynamic DoSomething(dynamic inputObject)

我可以使用枚举inputObject来调用它:

MyEnum myEnum;
myEnum = DoSomething(myEnum);

但出于某种原因,如果该函数将inputObject声明为ref dynamic类型而不是dynamic

dynamic DoSomething(ref dynamic inputObject)

由于无效转换,以下内容无效:

MyEnum myEnum;
DoSomething(ref myEnum);

Enums是否有一些特殊内容阻止我使用ref dynamic

2 个答案:

答案 0 :(得分:3)

作为引用传递的唯一方法是将myEnum转换为动态类型,然后通过引用传递它。我认为我们应该仔细研究生成的IL,以了解幕后发生的事情。让我们发现原因并分析这个程序:

 enum MyEnum{
     A,B
}

void Main()
{
    MyEnum  myEnum = MyEnum.B; //Assign a variable

    DoSomethingByEnum(myEnum); //Pass myEnum
    DoSomethingDynamicByValue(myEnum); //pass myEnum to a dynamic parameter

    dynamic dyn = myEnum;      //assign myenum to a dynamic variable
    DoSomethingDynamicByRef(ref dyn);    //pass it as a reference
}

MyEnum DoSomethingByEnum(MyEnum a)
{
   return a;
}

dynamic DoSomethingDynamicByValue(dynamic inputObject)
{
   return inputObject;

}

dynamic DoSomethingDynamicByRef(ref  dynamic inputObject)
{
  return inputObject;

} 

首先,我们调用DoSomethingByEnum按值传递变量myEnum,然后再调用DoSomethingDynamicByValue传递myEnum,但隐式地将其作为动态类型进行装箱。这是在MSIL级别发生的事情:

Main:
  IL_0001:  ldc.i4.1    // MyEnum myEnum = MyEnum.B;
  IL_0002:  stloc.0     // myEnum popped from evaluation stack and stored in a local variable 
  IL_0003:  ldarg.0     
  IL_0004:  ldloc.0     // myEnum loaded from local variable at index 0 and passed to the function
  IL_0005:  call        DoSomethingByEnum
  IL_000A:  pop         
  IL_000B:  ldarg.0     
  IL_000C:  ldloc.0     // myEnum
  IL_000D:  box         MyEnum // dynamic dyn = myEnum;  
                       // myEnum Converted from value type to a true object reference of type dynamic
  IL_0012:  call        DoSomethingDynamicByValue

DoSomethingByEnum(MyEnum)和DoSomethingDynamicByValue(动态)之间的唯一区别是装入变量myEnum(通过创建新对象并将值类型中的数据复制到新分配的动态对象中来实现)。看看Box Opcode

让我们来看看DoSomethingByEnum(MyEnum)和DoSomethingDynamicByValue(动态)IL:

 DoSomethingDynamicByValue:
    IL_0000:  nop         
    IL_0001:  ldarg.1     
    IL_0002:  stloc.0      
    IL_0003:  br.s        IL_0005
    IL_0005:  ldloc.0      
    IL_0006:  ret         

DoSomethingByEnum:
    IL_0000:  nop         
    IL_0001:  ldarg.1     
    IL_0002:  stloc.0      
    IL_0003:  br.s        IL_0005
    IL_0005:  ldloc.0     
    IL_0006:  ret         

无论变量的类型如何,两个函数的IL代码都完全相同。我们甚至可以拥有任何对象类型,但变量传递和共享调用的方式不会改变。

让我们看看DoSomethingDynamicByRef(ref动态)中发生了什么。

继续主要方法

Main:
    IL_0018:  ldloc.0     // myEnum
    IL_0019:  box         UserQuery.MyEnum
    IL_001E:  stloc.1     // dyn
    IL_001F:  ldarg.0     
    IL_0020:  ldloca.s    01 // loads the address of dyn onto the stack
    IL_0022:  call        UserQuery.DoSomethingDynamicByRef

DoSomethingDynamicByRef:
   IL_0000:  nop         
   IL_0001:  ldarg.1     
   IL_0002:  ldind.ref   //
   IL_0003:  stloc.0     
   IL_0004:  br.s        IL_0006
   IL_0006:  ldloc.0      
   IL_0007:  ret

此IL与前两个示例之间的区别依赖于这两个指令来加载和获取地址:

   ldloca.s    01  // loads the address of dyn onto the stack

   ldind.ref       // Loads the object reference at address addr onto the stack as a type O

我认为在ldloca.sldind.ref

上面的两条IL指令的MSDN页面中解释了无法传递不同对象类型地址的原因。
  

正确形成的Microsoft中间语言(MSIL)确保了这一点   ldind指令以与类型一致的方式使用   指针。最初推入堆栈的地址必须是   与机器上物体的自然尺寸对齐

希望这可以澄清一下。

答案 1 :(得分:1)

编译后

dynamic实际上是object,所以你实际上在问你为什么不能这样做:

void DoSomething(ref object input);

MyEnum myEnum;
DoSomething(ref myEnum);

原因是,ref不能像这样使用,请考虑下面的情况如何违反类型安全:

void DoSomething(ref object input) {
    input = new object();
}

正如Eric在评论中提到的那样,Enum在这里并不特别。