0

我正在为我关于 SO 的另一个问题尝试一些测试代码。

应该复制的代码:

(a, z) => a * b - Math.Log(z * b);

编码:

    static Func<int, int, double> IL_EmbedConst(int b)
    {
        var method = new DynamicMethod("EmbedConstIL", typeof(double), new[] { typeof(int), typeof(int) });

        var log = typeof(Math).GetMethod("Log", new Type[] { typeof(double) });

        var il = method.GetILGenerator();

        il.Emit(OpCodes.Ldarg_0);
        il.Emit(OpCodes.Ldc_I4, b);
        il.Emit(OpCodes.Mul);
        il.Emit(OpCodes.Conv_R8, b);

        il.Emit(OpCodes.Ldarg_1);
        il.Emit(OpCodes.Ldc_I4, b);
        il.Emit(OpCodes.Mul);
        il.Emit(OpCodes.Conv_R8, b);

        il.Emit(OpCodes.Call, log);

        il.Emit(OpCodes.Sub);

        il.Emit(OpCodes.Ret);

        return (Func<int, int, double>)method.CreateDelegate(typeof(Func<int, int, double>));
    }

使用:

        var mul1 = IL_EmbedConst(5);
        double res = mul1(4,6);

抛出:

Operation could destabilize the runtime.

看不出有什么问题(可能是因为我上次使用类似 asm 的语言是 25 年前)

4

1 回答 1

3

问题是您转换到的两个地方double

il.Emit(OpCodes.Conv_R8, b);

来自MSDN

OpCodes.Conv_R8 字段 将评估堆栈顶部的值转换为 float64。

该操作码不带任何参数。相反,只需使用:

il.Emit(OpCodes.Conv_R8);

代码运行良好并产生 16.5988026183378 作为输出。

于 2012-06-17T09:31:25.637 回答