发出断开连接的上下文错误



我试图在IL发出setter,但这个setter必须只接受不为空的对象,如果不是,它必须抛出异常。这是我的类Notnull,它接收一个il生成器和一个FieldBuilder,其中setter(如果不是null)必须存储其值。

 public class NonNull 
{
    public void CreateIL(ILGenerator il, FieldBuilder field)
    {
        il.Emit(OpCodes.Ldarg_0);        // Push "this" on the stack
        il.Emit(OpCodes.Newobj, typeof(Utils).GetConstructor(new Type[] { }));
        il.Emit(OpCodes.Stloc_1);
        il.Emit(OpCodes.Ldarg_1);
        il.Emit(OpCodes.Stloc_2);
        il.Emit(OpCodes.Ldloc_2);
        il.Emit(OpCodes.Ldloc_1);        // Push "value" on the stack
        il.Emit(OpCodes.Callvirt,typeof(Utils).GetMethod("checkIfNull"));
        il.Emit(OpCodes.Pop);
        il.Emit(OpCodes.Ldloc_2);
        il.Emit(OpCodes.Stfld, field);   // Set the field "_Name" to "value"
        il.Emit(OpCodes.Ret);
    }
}

类utils,我在IL中调用它并抛出异常。

class Utils
    {
        public void checkIfNull(object obj)
        {
            if (obj == null) throw new MyException();
        }
    }

当我尝试创建我的新类型(它有这个IL)时,它呈现一个断开连接的上下文错误。

使用非静态方法使它变得比它需要的更复杂。如果不需要对象引用来调用该方法,则该方法应该始终是静态的。

这就是你现在想要释放的东西:

public string Name
{
    set
    {
        Utils u = new Utils();
        string s = value;
        u.checkIfNull(s);
        _Name = s;
    }
}

绝对不需要创建Utils类的实例来进行检查。

首先将checkIfNull()设置为静态:

static class Utils
{
    public static void checkIfNull(object obj)
    {
        if (obj == null) throw new MyException();
    }
}

这是你应该尝试发出的:

public string Name
{
    set
    {
        Utils.checkIfNull(value);
        _Name = value;
    }
}

方法如下:

public class NonNull 
{
    public static void CreateIL(ILGenerator il, FieldBuilder field)
    {
        il.Emit(OpCodes.Ldarg_1); // Push "value" on the stack
        il.Emit(OpCodes.Call, typeof(Utils).GetMethod("checkIfNull"));
        il.Emit(OpCodes.Ldarg_0);
        il.Emit(OpCodes.Ldarg_1);
        il.Emit(OpCodes.Stfld, field);   // Set the field "_Name" to "value"
        il.Emit(OpCodes.Ret);
    }
}

我不明白你到底想做什么(与NotNull类和FieldBuilder),但我可以告诉你你需要你的setter属性(检查null和set字段)。

如果它对你有效,很好。如果没有,让我知道问题所在,我们会解决它。

首先让我们描述一下我们想要什么。

  1. 创建Utils
  2. 加载value参数
  3. 呼叫checkIfNull
  4. 加载this
  5. Load value
  6. 设置字段
  7. Ret

发出将是这样的:

il.Emit(OpCodes.Newobj, typeof(Utils).GetConstructor(new Type[] { }));
il.Emit(OpCodes.Ldarg_1);
il.Emit(OpCodes.Callvirt,typeof(Utils).GetMethod("checkIfNull"));
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Ldarg_1);
il.Emit(OpCodes.Stfld, field);
il.Emit(OpCodes.Ret);

结果应该是这样的:

newobj instance void MyApplication.Utils::.ctor() //create `Utils` object
ldarg.1                                           //load `value`
callvirt instance void 
         MyApplication.Utils::checkIfNull(object) //call checkForNull
ldarg.0                                           //load `this`
ldarg.1                                           //load `value`
stfld int32 MyApplication.Program::_field         //store `value` in `_field`
ret

相关内容

  • 没有找到相关文章

最新更新