7

私は次のILを一般化しようとしています(Reflectorから):

.method private hidebysig instance void SetValue(valuetype Test.TestFixture/ValueSource& thing, string 'value') cil managed
{
    .maxstack 8
    L_0000: nop 
    L_0001: ldarg.1 
    L_0002: ldarg.2 
    L_0003: call instance void Test.TestFixture/ValueSource::set_Value(string)
    L_0008: nop 
    L_0009: ret 
}

ただし、DynamicMethodを使用してこのILを再現しようとすると、次のようになります。

        [Test]
    public void Test_with_DynamicMethod()
    {
        var sourceType = typeof(ValueSource);
        PropertyInfo property = sourceType.GetProperty("Value");

        var setter = property.GetSetMethod(true);
        var method = new DynamicMethod("Set" + property.Name, null, new[] { sourceType.MakeByRefType(), typeof(string) }, true);
        var gen = method.GetILGenerator();

        gen.Emit(OpCodes.Ldarg_1); // Load input to stack
        gen.Emit(OpCodes.Ldarg_2); // Load value to stack
        gen.Emit(OpCodes.Call, setter); // Call the setter method
        gen.Emit(OpCodes.Ret);

        var result = (SetValueDelegate)method.CreateDelegate(typeof(SetValueDelegate));

        var source = new ValueSource();

        result(ref source, "hello");

        source.Value.ShouldEqual("hello");
    }

    public delegate void SetValueDelegate(ref ValueSource source, string value);

「操作によってランタイムが不安定になる可能性がある」という例外が発生します。ILは私と同じように見えますが、何かアイデアはありますか?ValueSourceは値型であるため、ここでrefパラメーターを実行しています。

編集

ValueSourceタイプは次のとおりです。

public struct ValueSource
{
   public string Value { get; set; }
}
4

1 に答える 1

4

引数を0/1(1/2ではなく)に変更します。

    gen.Emit(OpCodes.Ldarg_0); // Load input to stack
    gen.Emit(OpCodes.Ldarg_1); // Load value to stack

動的メソッドはインスタンスではなく静的として作成されているように見えるため(元のメソッドはインスタンスです)、したがって、引数は1つずれています。

(元の間違った答えでごめんなさい-あなたはコードの他のビットをそのままにしておくことができますtrue

于 2009-08-14T13:36:35.063 に答える