如何从动态发出的程序集中获得有意义的异常?

时间:2013-03-19 00:59:20

标签: reflection f#

尝试调试发出错误,是否有一种简单的方法可以找到有关异常的更多信息c 由发出的代码引起的?

例如,使用以下代码:

let dynamicAssembly =
    let asmName = new AssemblyName("MyAsm")
    let asmBuilder = AssemblyBuilder.DefineDynamicAssembly(asmName, AssemblyBuilderAccess.Run)
    let moduleBuilder = asmBuilder.DefineDynamicModule("MyModule")
    let typeBuilder = moduleBuilder.DefineType("MyDynamicType")
    let methodBuilder = 
        let build = typeBuilder.DefineMethod("MyMethod", MethodAttributes.Public, 
                                                CallingConventions.Standard,
                                                typeof<Int32>,
                                                [|typeof<Int32>; typeof<Int32>|])
        let ilGen = build.GetILGenerator()
        ilGen.Emit(OpCodes.Ldarg_0)
        ilGen.Emit(OpCodes.Ldarg_1)
        ilGen.Emit(OpCodes.Add)
        ilGen.Emit(OpCodes.Ret)

    typeBuilder.CreateType() |> ignore
    asmBuilder

let myType = dynamicAssembly.GetType("MyDynamicType")
let myObj = Activator.CreateInstance(myType)
myObj.GetType().GetMethod("MyMethod").Invoke(myObj, [|2; 3|])  |> ignore

我得到了“调用目标引发了异常。当我尝试在倒数第二行调用Invoke时。发送代码一直存在问题,但如果我可能不那么痛苦可以弄清楚如何获得有意义的例外。想法?

1 个答案:

答案 0 :(得分:5)

获得无用的异常是生成IL代码的乐趣的一部分。当你得到“调用目标抛出了异常”时,你可以查看InnerException属性,看看你是否可以在那里获得更多有用的东西:

try
  myObj.GetType().GetMethod("MyMethod").Invoke(myObj, [|2; 3|])  |> printfn "%A"
with 
  e -> printfn "%A" e.InnerException

在这种情况下,内部异常说:

  

System.InvalidProgramException:公共语言运行时检测到无效程序。

这只是意味着您生成的IL错误 - 我担心您不会收到更好的错误消息,但您可以将生成的程序集保存到磁盘并运行peverify或Reflector / ILSpy来查看他们将如何解释生成的代码。这意味着您需要添加另一个代码路径来生成正确的(非动态)程序集,但我认为这是值得的 - 您需要经常调试生成的IL ...

在这种情况下,问题是Ldarg_0引用this(而不是第一个参数),因此您需要生成:

ilGen.Emit(OpCodes.Ldarg_1)
ilGen.Emit(OpCodes.Ldarg_2)
ilGen.Emit(OpCodes.Add)
ilGen.Emit(OpCodes.Ret)