- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我有一个 WPF 控件,文档很差。
在代码隐藏中,我想反射(reflect)控件使用 GetType().GetEVents()
触发的事件,并为每个事件添加一个处理程序,它只打印出事件的名称.
这将使我能够看到与控件的交互实际上在做什么。
到目前为止我有:
foreach (var e in GetType().GetEvents())
{
var name = e.Name;
var handler = new Action<object,object>( (o1,o2) =>Console.WriteLine(name));
try
{
e.AddEventHandler(
this,
Delegate.CreateDelegate(
e.EventHandlerType,
handler.Target,
handler.Method
));
}
catch (Exception ex)
{
Console.WriteLine( "Failed to bind to event {0}", e.Name);
}
}
这似乎在事件签名为 (object,EventArgs)
时有效,但在某些其他事件上无法绑定(bind)。
有没有办法在不知道事件签名的情况下做到这一点?
最佳答案
您可以使用 System.Linq.Expressions.Expression
生成与事件签名匹配的动态处理程序的类 - 您只需在其中调用 Console.WriteLine
.
Expression.Lambda
方法(已提供指向您需要的特定重载的链接)可用于生成 Func<>
或者,更有可能的是 Action<>
类型正确。
您反射(reflect)事件的委托(delegate)类型(如@Davio 所述,获取它的 Invoke
方法)以提取所有参数并创建 ParameterExpression
s 用于提供给 lambda 方法的每一个。
这是一个完整的解决方案,您可以将其粘贴到标准单元测试中,我将在之后的后续编辑中进行解释:
public class TestWithEvents
{
//just using random delegate signatures here
public event Action Handler1;
public event Action<int, string> Handler2;
public void RaiseEvents(){
if(Handler1 != null)
Handler1();
if(Handler2 != null)
Handler2(0, "hello world");
}
}
public static class DynamicEventBinder
{
public static Delegate GetHandler(System.Reflection.EventInfo ev) {
string name = ev.Name;
// create an array of ParameterExpressions
// to pass to the Expression.Lambda method so we generate
// a handler method with the correct signature.
var parameters = ev.EventHandlerType.GetMethod("Invoke").GetParameters().
Select((p, i) => Expression.Parameter(p.ParameterType, "p" + i)).ToArray();
// this and the Compile() can be turned into a one-liner, I'm just
// splitting it here so you can see the lambda code in the Console
// Note that we use the Event's type for the lambda, so it's tightly bound
// to that event.
var lambda = Expression.Lambda(ev.EventHandlerType,
Expression.Call(typeof(Console).GetMethod(
"WriteLine",
BindingFlags.Public | BindingFlags.Static,
null,
new[] { typeof(string) },
null), Expression.Constant(name + " was fired!")), parameters);
//spit the lambda out (for bragging rights)
Console.WriteLine(
"Compiling dynamic lambda {0} for event \"{1}\"", lambda, name);
return lambda.Compile();
}
//note - an unsubscribe might be handy - which would mean
//caching all the events that were subscribed for this object
//and the handler. Probably makes more sense to turn this type
//into an instance type that binds to a single object...
public static void SubscribeAllEvents(object o){
foreach(var e in o.GetType().GetEvents())
{
e.AddEventHandler(o, GetHandler(e));
}
}
}
[TestMethod]
public void TestSubscribe()
{
TestWithEvents testObj = new TestWithEvents();
DynamicEventBinder.SubscribeAllEvents(testObj);
Console.WriteLine("Raising events...");
testObj.RaiseEvents();
//check the console output
}
大纲——我们从一个有一些事件的类型开始(我正在使用 Action
但它应该适用于任何东西),并且有一个我们可以用来测试触发所有那些有订阅者的事件的方法。
然后到DynamicEventBinder
类,它有两个方法:GetHandler
- 为特定类型的特定事件获取处理程序;和 SubscribeAllEvents
它为该类型的给定实例绑定(bind)所有这些事件——它简单地遍历所有事件,调用 AddEventHandler
对于每个,调用 GetHandler
获取处理程序。
GetHandler
方法是核心所在 - 并且完全按照我在大纲中的建议进行。
委托(delegate)类型有一个 Invoke
成员由编译器编译到其中,它反射(reflect)了它可以绑定(bind)到的任何处理程序的签名。因此,我们反射(reflect)该方法并获取它具有的任何参数,创建 Linq ParameterExpression
每个实例。命名参数很巧妙,重要的是这里的类型。
然后我们构建一个单行 lambda,其主体基本上是:
Console.WriteLine("[event_name] was fired!");
(注意这里事件的名称被拉入动态代码并合并到一个常量字符串中,就代码而言)
当我们构建 lambda 时,我们还告诉 Expression.Lambda
方法我们打算构建的委托(delegate)类型(直接绑定(bind)到事件的委托(delegate)类型),并通过传递 ParameterExpression
我们之前创建的数组,它会生成一个有那么多参数的方法。我们使用 Compile
实际编译动态代码的方法,它给我们一个 Delegate
然后我们可以将其用作 AddEventHandler
的参数.
我真诚地希望这能解释我们所做的事情 - 如果您在表达式和动态代码成为令人费解的东西之前没有使用过它。事实上,与我共事的一些人只是简单地称之为巫术。
关于c# - 使用反射动态绑定(bind)到 C# 事件,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13083126/
一、反射 1.定义 Java的反射(reflection)机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法(即使是私有的);对于任意一个对象,都能够调用它的任意方法和属性,那么,我
有没有办法从 JavaScript 对象内部获取所有方法(私有(private)、特权或公共(public))?这是示例对象: var Test = function() { // private m
我有一个抽象类“A”,类“B”和“C”扩展了 A。我想在运行时根据某些变量创建这些实例。如下所示: public abstract class A { public abstract int
假设我们在内存中有很多对象。每个都有一个不同的ID。如何迭代内存以找到与某些 id 进行比较的特定对象?为了通过 getattr 获取并使用它? 最佳答案 您应该维护这些对象的集合,因为它们是在类属性
假设我有这个结构和一个方法: package main import ( "fmt" "reflect" ) type MyStruct struct { } func (a *MyS
C#反射简介 反射(Reflection)是C#语言中一种非常有用的机制,它可以在运行时动态获取对象的类型信息并且进行相应的操作。 反射是一种在.NET Framework中广
概述 反射(Reflection)机制是指在运行时动态地获取类的信息以及操作类的成员(字段、方法、构造函数等)的能力。通过反射,我们可以在编译时期未知具体类型的情况下,通过运行时的动态
先来看一段魔法吧 public class Test { private static void changeStrValue(String str, char[] value) {
结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套; go中的struct类型理解为类,可以定义方法,和函数定义有些许区别; struct类型是值类型
反射 1. 反射的定义 Java的反射(reflection)机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性,既然能拿到那么,我们
反射的定义 java的反射(reflection) 机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性,既然能拿到嘛,那么,我们就可以
我有一个 Java POJO: public class Event { private String id; private String name; private Lon
我编写了以下函数来检查给定的单例类是否实现了特征。 /** Given a singleton class, returns singleton object if cls implements T.
我正在研究 Java 反射的基础知识并观察有关类方法的信息。我需要获得一个符合 getMethod() 函数描述的规范的方法。然而,当我这样做时,我得到了一个 NoSuchMethodExceptio
我正在通过以下代码检索 IEnumerable 属性列表: BindingFlags bindingFlag = BindingFlags.Instance | BindingFlags.Public
我需要检查属性是否在其伙伴类中定义了特定属性: [MetadataType(typeof(Metadata))] public sealed partial class Address { p
我正在尝试使用 Reflections(由 org.reflections 提供)来处理一些繁重的工作,因此我不需要在很长的时间内为每个类手动创建一个实例列表。但是,Reflections 并未按照我
scala 反射 API (2.10) 是否提供更简单的方法来搜索加载的类并将列表过滤到实现定义特征的特定类? IE; trait Widget { def turn(): Int } class
我想在运行时使用反射来查找具有给定注释的所有类,但是我不知道如何在 Scala 中这样做。然后我想获取注释的值并动态实例化每个映射到关联注释值的带注释类的实例。 这是我想要做的: package pr
这超出了我的头脑,有人可以更好地向我解释吗? http://mathworld.wolfram.com/Reflection.html 我正在制作一个 2d 突破格斗游戏,所以我需要球能够在它击中墙壁
我是一名优秀的程序员,十分优秀!