IL,发出默认构造函数调用

时间:2011-07-27 13:12:23

标签: c# .net code-generation il

我在运行时生成新类型, 在我生成默认构造函数之后,我想生成另一个带参数的函数。我这样做:

cb = tb.DefineConstructor(MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.RTSpecialName,
                    CallingConventions.Standard, new Type[] { typeof(bool) });
GenConstructorWithParameters(cb, fields, genFields);

问题是,我无法从方法GenConstructorWithParameters调用默认构造函数,因为CLR不允许我写这样的东西:

gen.Emit(OpCodes.Ldarg_0);        
gen.Emit(OpCodes.Call, cb.DeclaringType.GetConstructor(Type.EmptyTypes));//Not allowed to call .GetConstructor() on not created type!

如何发出默认构造函数的调用?它有可能吗?

tb - TypeBuilder的实例,cb - ConstructorBuilder

3 个答案:

答案 0 :(得分:4)

您想要newobj

gen.Emit(OpCodes.Newobj, cb.DeclaringType.GetConstructor(Type.EmptyTypes));

答案 1 :(得分:4)

您应该将当前DeclaringType.GetConstructor传递给默认构造函数,而不是使用ConstructorBuilder

基本上,在构建类型时,在可能对现有类型使用基于反射的方法的地方,您应该传入已经使用过的构建器。


所以它会是:

gen.Emit(OpCodes.Call, defaultCB);

其中defaultCB是您在定义默认构造函数时声明的ConstructorBuilder

答案 2 :(得分:0)

Dim objCtor As ConstructorInfo = objType.GetConstructor(New Type() {})

      Dim pointCtor As ConstructorBuilder = pointTypeBld.DefineConstructor(MethodAttributes.Public, CallingConventions.Standard, ctorParams)
      Dim ctorIL As ILGenerator = pointCtor.GetILGenerator()

      ctorIL.Emit(OpCodes.Ldarg_0)
      ctorIL.Emit(OpCodes.Call, objCtor)

      ctorIL.Emit(OpCodes.Ldarg_0)
      ctorIL.Emit(OpCodes.Ldarg_1)
      ctorIL.Emit(OpCodes.Stfld, xField)

      ctorIL.Emit(OpCodes.Ldarg_0)
      ctorIL.Emit(OpCodes.Ldarg_2)
      ctorIL.Emit(OpCodes.Stfld, yField)

      ctorIL.Emit(OpCodes.Ldarg_0)
      ctorIL.Emit(OpCodes.Ldarg_3)
      ctorIL.Emit(OpCodes.Stfld, zField)

      ctorIL.Emit(OpCodes.Ret)

Dim myDynamicType As Type = Nothing
Dim myDTctor As ConstructorInfo = myDynamicType.GetConstructor(aPtypes)
      Console.WriteLine("Constructor: {0};", myDTctor.ToString())