2010-04-07 1 views
0

Может кто-нибудь объяснить мне, почему следующее работает для первого теста, но генерирует исключение InvalidProgramException для второго теста? Я в тупике.Ошибка компилятора JIT - Неверное исключение программы с использованием Reflection.Emit

using System; 
using System.Reflection; 
using System.Reflection.Emit;

namespace DMTest { class Program { static void Main(string[] args) { Test.NewResultWithParam(); Console.WriteLine("Press any key to continue"); Console.ReadLine(); Test.NewResult(); Console.WriteLine("Press any key to exit"); Console.ReadLine(); } }

public static class Test { public static void NewResult() { var f = typeof(Result).New<Result>(); var result = f.Invoke(); } public static void NewResultWithParam() { var f = typeof(Result).New<Param, Result>(); var result = f.Invoke(new Param("Parameter Executed")); result.Build(); } } public static class DynamicFunctions { public static Func<R> New<R>(this Type type) { if (!typeof(R).IsAssignableFrom(type)) { throw new ArgumentException(); } var dm = BuildNewMethod(type, Type.EmptyTypes); return (Func<R>)dm.CreateDelegate(typeof(Func<R>)); } public static Func<T, R> New<T, R>(this Type type) { if (!typeof(R).IsAssignableFrom(type)) { throw new ArgumentException(); } var dm = BuildNewMethod(type, new Type[] { typeof(T) }); return (Func<T, R>)dm.CreateDelegate(typeof(Func<T, R>)); } private static DynamicMethod BuildNewMethod(Type newObjType, Type[] parameterTypes) { var dm = new DynamicMethod("New_" + newObjType.FullName, newObjType, parameterTypes, newObjType); var info = newObjType.GetConstructor(BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public, null, parameterTypes, new ParameterModifier[0]); ILGenerator il = dm.GetILGenerator(); il.Emit(OpCodes.Ldarg_0); il.Emit(OpCodes.Newobj, info); il.Emit(OpCodes.Ret); return dm; } } public class Result { private Param _param; private Result() { Console.WriteLine("Parameterless constructor called"); } private Result(Param param) { Console.WriteLine("Parametered constructor called"); _param = param; } public void Build() { _param.Execute(); } } public class Param { private string _s; public Param(string s) { _s = s; } public void Execute() { Console.WriteLine(_s); } }

}

ответ

3
il.Emit(OpCodes.Ldarg_0); // <-- problem here 

Поскольку ваш динамический метод не имеет параметров в версии New<T>, вы не можете использовать Ldarg_0. Кроме того, Newobj ожидает нулевые аргументы, поэтому ваш стек несбалансирован.

+0

Исправлено. Спасибо. Я подумал, что это было что-то простое, что мне не хватало. –

 Смежные вопросы

  • Нет связанных вопросов^_^