Генерация динамических событий в C# с использованием DynamicMethod и ILGenerator
-
21-08-2019 - |
Вопрос
Мне нужно создать обработчик событий на основе объекта EventInfo во время выполнения и вызвать метод в этом обработчике событий.Что-то вроде следующего:
public void RegisterAction(ActionData actionData, EventInfo eventInfo,
Control control)
{
MethodInfo methodInfo = eventInfo.EventHandlerType.GetMethod("Invoke");
List<Type> ps = new List<Type>();
foreach (ParameterInfo info in methodInfo.GetParameters())
{
ps.Add(info.ParameterType);
}
DynamicMethod method = new DynamicMethod("Adapter",
typeof (void),
ps.ToArray(),
GetType(),
true);
ILGenerator generator = method.GetILGenerator();
// Here I need to generate a method to do the following:
// ExecuteAction(actionData);
// Then I can use this runtime method as an event handler and
// bind it to the control
Delegate proxy = method.CreateDelegate(eventInfo.EventHandlerType, this);
eventInfo.AddEventHandler(control, proxy);
}
Мне нужна помощь в создании IL-кода для комментируемой части.
Решение
public void RegisterAction(ActionData actionData, EventInfo eventInfo,
Control control)
{
MethodInfo methodInfo = eventInfo.EventHandlerType.GetMethod("Invoke");
List<Type> ps = new List<Type>();
ps.Add (typeof (ActionData)) ;
foreach (ParameterInfo info in methodInfo.GetParameters())
{
ps.Add(info.ParameterType);
}
DynamicMethod method = new DynamicMethod("Adapter",
typeof (void),
ps.ToArray(),
GetType(),
true);
// compatible signatures for ExecuteAction
// (assuming you aren't interested in sender and eventArgs):
// static void ExecuteAction (ActionData) ;
// void ActionData.ExecuteAction () ;
MethodInfo miExecuteAction = <...> ;
ILGenerator generator = method.GetILGenerator();
generator.Emit (OpCodes.Ldarg_0) ;
generator.Emit (OpCodes.Call, miExecuteAction) ;
generator.Emit (OpCodes.Ret) ;
// if you want to pass this to ExecuteAction,
// you'll need to put it into actionData.
Delegate proxy = method.CreateDelegate(eventInfo.EventHandlerType, actionData);
eventInfo.AddEventHandler(control, proxy);
}
Редактировать:если подумать, если все ваши события следуют шаблону (sender, args), вам даже не нужно возиться с SRE:
public static void Execute<T> (ActionData data, object sender, T args)
where T : EventArgs
{
ExecuteAction (data) ;
}
public void RegisterAction (ActionData actionData, EventInfo eventInfo,
Control control)
{
MethodInfo compatibleMethod = typeof (this).GetMethod ("Execute",
BindingFlags.Static | BindingFlags.Public).MakeGenericMethod (
eventInfo.EventHandlerType.GetMethod ("Invoke").GetParameters ()[1].ParameterType)) ;
eventInfo.AddEventHandler (control,
Delegate.CreateDelegate (eventInfo.EventHandlerType, actionData,
compatibleMethod)) ;
}
Другие советы
Я предполагаю, что вы хотите сделать делегата из MethodInfo вашего события...
Если так, вот статья описывающее, как это сделать.Необходимые коды IL описаны в этой статье.
Не связан с StackOverflow