一般來說,接口只能在編碼時實現,但由於某些原因,無法在運行時實現。下面就介紹一種在運行時實現接口的解決方案。
在介紹之前,先學習這個方案中所需要的知識點是動態編譯、委托和反射。
分析:
1、實現接口還是必須有一個類,一個“class ClassName : InterfaceName {...}”的類。順着這個思路,會想到這個ClassName能否動態生成(運行時生成)。
2、動態生成類,有兩種方式:動態編譯和Emit技術(這里使用“動態編譯”)。如果每實現一個類都需要動態編譯,這樣一定承擔上影響性能,也占內存。所以思考這個類能不能是一個“模版類”,一個接口對應一個“動態實現的模版類”(當然,不同接口肯定是對應不同的“動態實現的模版類”),不同的實現,由你自己決定。
3、“動態實現的模版類”會想到委托。如:
public class ClassNameImplementTemplate : InterfaceName { private Dictionary<string, Delegate> _implements; // 具體實現的委托
// 實現接口的方法
public void Func()
{
// 這里執行_implements["Func"]的Delegate
}
}
在“動態實現的模版類(ClassNameImplementTemplate)”里維護一個具體實現列表,也就是一個委托列表。在“動態實現的模版類”中的實現接口的方法里,去對應執行這個委托列表的具體實現委托。
通過上面的分析,總結思路,步驟如下:
1、通過“動態實現接口”生成“動態實現的模版類”。
2、用戶具體實現屬性、方法、事件等。
3、用戶實例化“動態實現的模版”對象。
按照上次的步驟,我們對其進行封裝:RuntimeClass<Interface>類。類圖如下:

該類包含了生成“動態實現的模版類”、用戶實現方法、用戶實現屬性、用戶觸發事件(可在實現屬性或方法中觸發事件)以及最后的實例化。當用戶需要動態實現接口ITry時,可用下面代碼:
public interface ITry { string Prop1 { get; set; } event EventHandler Click; string Fun2(); } RuntimeClass<ITry> cls = new RuntimeClass<ITry>(); cls.ImplementProperty<string>("Prop1", // 具體實現Prop1屬性 (@this) => // @this是實例本身,這樣即可調用實例的其他方法或訪問屬性或觸發事件 {
// 相當於屬性的get return cls.Fields["prop1"].ToString(); // 字符 }, (@this, s) => {
// 相當於屬性的set cls.Fields["prop1"] = s; }).ImplementMethod("Fun2", new Func<ITry, string>((@this) => // 具體實現Fun2方法 { cls.OnEventHandler(@this, "Click", null, EventArgs.Empty); //觸發事件 @this.Prop1 = "3212"; return @this.Prop1; })
); ITry obj = (ITry)cls.CreateInstance();// 實例化 obj.Click += (sender, e) => // 注冊事件 { MessageBox.Show("點擊"); }; obj.Prop1 = "123"; // 屬性賦值 string ss = obj.Prop1; // 獲取屬性 ss = obj.Fun2(); // 執行方法
VS2010,.net4 代碼下載: http://download.csdn.net/detail/lyb018/4685241
通過這種解決方案,可以擴展出動態代理、攔截器等等。
