我试图在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字段)。
如果它对你有效,很好。如果没有,让我知道问题所在,我们会解决它。
首先让我们描述一下我们想要什么。
- 创建
Utils
类 - 加载
value
参数 呼叫 - 加载
this
- Load
value
- 设置字段
- Ret
checkIfNull
发出将是这样的:
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