Ork*_*ane 4 c# constructor delegates reflection.emit dynamic-method
我正在尝试通过发出动态方法来创建构造函数的委托表示,该方法必须匹配这个非常"松散类型"的签名,以便它可以与任何类型的参数化构造函数一起使用:
public delegate Object ParamsConstructorDelegate(params object[] parameters);
Run Code Online (Sandbox Code Playgroud)
并且创建委托的代码看起来像(注意这是针对Silverlight的)
public static ParamsConstructorDelegate CreateDelegate(ConstructorInfo constructor)
{
Guard.ArgumentNotNull(constructor, "constructor");
Guard.ArgumentValue(constructor.GetParameters().Length == 0, MUSTBE_PARAMETERIZED_CONSTRUCTOR);
var _argumentTypes = new Type[] { typeof(object[]) };
var _parameters = constructor.GetParameters();
var _parameterTypes = _parameters.Select((p) => p.ParameterType).ToArray();
var _sourceType = constructor.DeclaringType;
var _method = new DynamicMethod(constructor.Name, _sourceType, _argumentTypes);
var _gen = _method.GetILGenerator();
for (var _i = 0; _i < _parameters.Length; _i++)
{
if (_parameters[_i].IsOut || _parameterTypes[_i].IsByRef)
{
if (_i < 128)
{
_gen.Emit(OpCodes.Ldarga_S, (byte)_i);
}
else
_gen.Emit(OpCodes.Ldarga, _i);
}
else
{
switch (_i)
{
case 0:
_gen.Emit(OpCodes.Ldarg_0, _i);
break;
case 1:
_gen.Emit(OpCodes.Ldarg_1, _i);
break;
case 2:
_gen.Emit(OpCodes.Ldarg_2, _i);
break;
case 3:
_gen.Emit(OpCodes.Ldarg_3, _i);
break;
default:
if (_i < 128)
_gen.Emit(OpCodes.Ldarg_S, (byte)_i);
else
_gen.Emit(OpCodes.Ldarg, _i);
break;
}
}
}
_gen.Emit(OpCodes.Newobj, constructor);
_gen.Emit(OpCodes.Ret); ;
return (ParamsConstructorDelegate)_method.CreateDelegate(typeof(ParamsConstructorDelegate));
}
Run Code Online (Sandbox Code Playgroud)
现在,我得到一个"操作可能会破坏运行时的稳定性".验证异常,显然IL错了,所以我希望有人可以纠正我.
谢谢
我可以看到两个问题; 首先,你不需要_i
为Ldarg_0
直通Ldarg_3
情况下(这是隐含的).其次-您的代理只能有一个ARG(阵列).你将需要从数组中获取项目并进行强制转换- 如下所示(仅处理按值传递;对于ref
/ out
您必须定义本地并使用stloc/ldloca/etc):
using System;
using System.Reflection.Emit;
public delegate object ParamsConstructorDelegate(params object[] parameters);
public class Foo
{
string s;
int i;
float? f;
public Foo(string s, int i, float? f)
{
this.s = s;
this.i = i;
this.f = f;
}
}
static class Program
{
static void Main()
{
var ctor = Build(typeof(Foo));
Foo foo1 = (Foo)ctor("abc", 123, null);
Foo foo2 = (Foo)ctor(null, 123, 123.45F);
}
static ParamsConstructorDelegate Build(Type type)
{
var mthd = new DynamicMethod(".ctor", type,
new Type[] { typeof(object[]) });
var il = mthd.GetILGenerator();
var ctor = type.GetConstructors()[0]; // not very robust, but meh...
var ctorParams = ctor.GetParameters();
for (int i = 0; i < ctorParams.Length; i++)
{
il.Emit(OpCodes.Ldarg_0);
switch (i)
{
case 0: il.Emit(OpCodes.Ldc_I4_0); break;
case 1: il.Emit(OpCodes.Ldc_I4_1); break;
case 2: il.Emit(OpCodes.Ldc_I4_2); break;
case 3: il.Emit(OpCodes.Ldc_I4_3); break;
case 4: il.Emit(OpCodes.Ldc_I4_4); break;
case 5: il.Emit(OpCodes.Ldc_I4_5); break;
case 6: il.Emit(OpCodes.Ldc_I4_6); break;
case 7: il.Emit(OpCodes.Ldc_I4_7); break;
case 8: il.Emit(OpCodes.Ldc_I4_8); break;
default: il.Emit(OpCodes.Ldc_I4, i); break;
}
il.Emit(OpCodes.Ldelem_Ref);
Type paramType = ctorParams[i].ParameterType;
il.Emit(paramType.IsValueType ? OpCodes.Unbox_Any
: OpCodes.Castclass, paramType);
}
il.Emit(OpCodes.Newobj, ctor);
il.Emit(OpCodes.Ret);
return (ParamsConstructorDelegate)
mthd.CreateDelegate(typeof(ParamsConstructorDelegate));
}
}
Run Code Online (Sandbox Code Playgroud)
对于信息 - 我很懒 - 如果我想知道要写什么IL我用C#写它然后加载到反射器中.例如,要做到这一点,我写了一个方法:
static object CreateFoo(object[] vals)
{
return new Foo((string)vals[0], (int)vals[1], (float?)vals[2]);
}
Run Code Online (Sandbox Code Playgroud)
并从那里扭转它