2012-11-20 117 views
4

我想使用反射发射来创建具有任意构造函数参数的类的实例。这是我的代码看起来像:反射发射创建类实例

public delegate object ObjectActivator(params object[] args); 
static void Main(string[] args) 
{ 
    var ao = new { ID = 10000, FName = "Sample", SName = "Name" }; 
    var t = ao.GetType(); 
    var info = t.GetConstructor(new Type[] { typeof(int), typeof(string), typeof(string) }); 
    var objActivatorEmit = GetActivatorEmit(info); 
    var obj = createdActivatorEmit(4, "Foo", "Bar"); 
} 
public static ObjectActivator GetActivatorEmit(ConstructorInfo ctor) 
{ 
    ParameterInfo[] paramsInfo = ctor.GetParameters(); 
    DynamicMethod method = new DynamicMethod("CreateInstance", typeof(object), new Type[] { typeof(object[]) }); 
    ILGenerator gen = method.GetILGenerator(); 
    for (int i = 0; i < paramsInfo.Length; i++) 
    { 
     Type t = paramsInfo[i].ParameterType; 
     gen.Emit(OpCodes.Ldarg_0); // Push array (method argument) 
     gen.Emit(OpCodes.Ldc_I4, i); // Push i 
     gen.Emit(OpCodes.Ldelem_Ref); // Pop array and i and push array[i] 
     if(t.IsValueType) 
     { 
      gen.Emit(OpCodes.Unbox_Any, t); // Cast to Type t 
     } 
     else 
     { 
      gen.Emit(OpCodes.Castclass, t); //Cast to Type t 
     } 
    } 
    gen.Emit(OpCodes.Newobj, ctor); 
    gen.Emit(OpCodes.Ret); 
    return (ObjectActivator)method.CreateDelegate(typeof(ObjectActivator)); 
} 

代码失败,并MethodAccessException并显示错误消息Attempt by method 'DynamicClass.CreateInstance(System.Object[])' to access method '<>f__AnonymousType1'3<System.Int32,System.__Canon,System.__Canon>..ctor(Int32, System.__Canon, System.__Canon)' failed.

什么问题?

回答

3

的错误信息表明匿名类型的构造是不公开的。我觉得匿名类型构造函数总是internal,所以你需要使用一个不同的DynamicMethod构造跳过可视性检查:

DynamicMethod method = new DynamicMethod("CreateInstance", typeof(object), new Type[] { typeof(object[]) }, true); 

注意,这不是在部分信任情况下工作。

+0

我认为这是一个通用的公共ctor,基于:http://blogs.msdn.com/b/ericlippert/archive/2010/12/20/why-are-anonymous-types-generic.aspx – Elisha

+0

@Elisha - 它是一个内部类型的公共构造函数,所以它仍然无法访问。 – kvb

+0

现在工作正常,谢谢。 – user1622959

2

你不需要使用Reflection.Emit,我建议你不要。除非你知道自己在做什么或者有某些其他API无法满足的特殊需求,否则最好不要离开。

有3个替代方案更容易得到正确的。检查出来:

using System; 
using System.Linq; 
using System.Linq.Expressions; 
using System.Reflection; 

public static class App 
{ 
    public delegate object ObjectActivator(params object[] args); 

    public static void Main(string[] args) 
    { 
     var ao = new { ID = 10000, FName = "Sample", SName = "Name" }; 
     var t = ao.GetType(); 
     var info = t.GetConstructor(new[] { typeof(int), typeof(string), typeof(string) }); 
     if (info == null) 
     { 
      throw new Exception("Info is null"); 
     } 

     // This uses System.Linq.Expressions to create the delegate 
     var activator = GetActivator(info); 
     var newObj1 = activator(4, "Foo", "Bar"); 

     // This invokes the ConstructorInfo directly 
     var newObj2 = info.Invoke(new object[] { 4, "Foo", "Bar" }); 

     // This uses System.Activator to dynamically create the instance 
     var newObj3 = Activator.CreateInstance(t, 4, "Foo", "Bar"); 
    } 

    // This uses System.Linq.Expressions to generate a delegate 
    public static ObjectActivator GetActivator(ConstructorInfo ctor) 
    { 
     var args = Expression.Parameter(typeof(object[]), "args"); 
     var parameters = ctor.GetParameters().Select((parameter, index) => Expression.Convert(Expression.ArrayIndex(args, Expression.Constant(index)), parameter.ParameterType)); 
     return Expression.Lambda<ObjectActivator>(Expression.New(ctor, parameters), args).Compile(); 
    } 
} 

注:灵感GetActivator方法从this post

+1

谢谢,但我正在写一个microbenchmark来测试所有选项,而Emit是唯一一个没有工作的人。 – user1622959

+0

我一直在努力与ILGenerator合作。使用'表达式'要容易得多。我用它缓存构造函数调用委托。非常感谢你 –