前言
通過放射,可以在運行時獲得.NET中每一個類型(包括類、結構、委托、接口和枚舉等)的成員,包括方法、屬性、事件,以及構造函數(shù)等。還可以獲得每個成員的名稱、限定符和參數(shù)等。有了反射,即可對每一個類型了如指掌。如果獲得了構造函數(shù)的信息,即可直接創(chuàng)建對象,即使這個對象的類型在編譯時還不知道。那么如何注冊事件呢?
本文將介紹如何使用反射注冊事件。下面話不多說了,來一起看看看詳細的介紹吧
不使用反射
例如,我們希望反射的類型是這樣的:
public class Walterlv { public event EventHandler BlogPublished; }
那么只需要使用如下代碼即可完成事件的注冊:
var walterlv = new Walterlv(); walterlv += Walterlv_BlogPublished;
public void Walterlv_BlogPublished(object sender, EventHandler handler) { }
使用反射
而如果使用反射,則是:
var walterlv = new Walterlv(); var eventInfo = typeof(Walterlv).GetEvent(nameof(BlogPublished)); var handler = new EventHandler(Walterlv_BlogPublished); eventInfo.AddEventHandler(walterlv, handler);
當然,實際使用的時候,如果能訪問到 Walterlv 類型,當然也不會去用到反射,所以通常情況是這樣的:
public void AddHandler<T>(T instance, string eventName, EventHandler handler) { var eventInfo = instance.GetType().GetEvent(eventName); eventInfo.AddEventHandler(instance, handler); }
安全地使用反射
雖然以上方式使用了反射成功注冊了事件,但實際上我們的參數(shù)中傳入了一個特定類型的委托 EventHandler。實際上事件的委托種類非常多。
在委托中,即便簽名完全相同,也不是同一個委托類型。如果傳入的參數(shù)類型改為 EventHandler<EventArgs>
,或者 BlogPublished 事件的類型改為 EventHandler<EventHandler>,雖然實際上這兩個委托的簽名是兼容的,但其委托類型不同,依然是不能互相轉換的。你會在運行時遇到一下異常:
▲ 委托無法轉換
所以我們必須有一些更安全的方式來注冊事件。
正常情況下,我們轉換一個簽名兼容的委托是使用構造函數(shù):
public EventHandler ConvertDelegate(EventHandler<EventArgs> handler) { return new EventHandler(handler); }
那么在反射中,我們需要使用 Delegate.CreateDelegate 創(chuàng)建指定類型的委托。
public void AddHandler<T>(T instance, string eventName) { var eventInfo = instance.GetType().GetEvent(eventName); var methodInfo = GetType().GetMethod(nameof(Walterlv_BlogPublished)); var @delegate = Delegate.CreateDelegate(eventInfo.EventHandlerType, this, methodInfo); eventInfo.AddEventHandler(instance, @delegate); } public void Walterlv_BlogPublished(object sender, EventHandler handler) { }
這里,Delegate.CreateDelegate
的作用就是執(zhí)行委托類型的轉換。我在 .NET Core/Framework 創(chuàng)建委托以大幅度提高反射調用的性能 中也提到過這個方法。
參考資料
c# - AddEventHandler using reflection - Stack Overflow
總結
聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com