什麼是插件式編程
提起插件式,我們首先想到的是firefox,用過firefox的人都知道它是一個插件式程序。當一個功能需要,完全可以從網上下載一個插件後,重啟後,就能使用。這個功能給我們帶來許多的方便之處,這就是插件式程序的好處。
插件的本質在於不修改程序主體(平台)的情況下對軟件功能進行拓展與加強,當插件的接口公開後,任何公司或個人都可以制作自己的插件來解決一些操作上的不便或增加新功能,也就是真正意義上實現“即插即用”軟件開發。
平台+插件軟件結構是將一個待開發的目標軟件分為兩部分,一部分為軟件的主體或框架,可定義為平台,這是預先編譯後的程序。另一部分為功能或補充模塊,可定義為插件。這個就是後來要進行安裝的插件程序。
假設你的程序已經部署在用戶的計算機上,並且能夠正常運行了。但是有一天,用戶打來電話——他們需要增加新的功能。確定了用戶的需求後,你竟然發現原有的軟件架構已經無法勝任新增任務的需求——你需要重新設計這個應用了!但問題是,就算你又用了一個開發周期完成了用戶需要的應用,切不能保證用戶的需求不會再次變更。也就是說,需求蔓延的可能性依然存在。因此,這種情況下插件架構更能顯示出它的優越性。
可以這麼說,用它可以帶來方便的地方,而且開發它,也很簡單。而且這樣的主程序根本就不需要改動。需要插件時,拿來就能用,插件更新時,也只需更新這個插件即可。
從程序開發這角度,一般是先開發主程序,決定哪些功能由主程序來完成,然後再建立接口,申明接口的內容,這些內容決定著插件功能的擴展,及方向的。這些都是有主程序開發者預先准備好的。插件開發者,從主程序開發者那裡得到接口的內容,並書寫繼承這些接口的類,來完成具體的功能。
下面來寫個例子,這個例子沒實際意義,純屬學習思想。例子是網上的經過自己改造的,發現別人某些地方不合理。
首先,新建一個類庫,裡面定義接口,這裡定義兩個方法,一個有返回值的,一個無返回值的。
using System;using System.Collections.Generic;using System.Text;namespace IMsg{ /// <summary> /// 這是插件必須實現的接口,也是主程序與插件通信的唯一接口 /// 換句話說,主程序只認識插件裡的這些方法 /// </summary> public interface IMsgPlug { void OnShowDlg(); string OnShowInfo(); }}
將上面的類庫生成IMsg.dll,新建一個類庫MYPlugin1,添加剛出的引用,分別新建兩個類來實現IMsg中定義的接口。
using System;
using System.Collections.Generic;using System.Text;using IMsg;namespace MYPlugin1{
public class myConsole : IMsgPlug
{
#region IMsgPlug 成員
public void OnShowDlg()
{
Console.WriteLine("控制台調用插件的OnShowDlg方法");
}
public string OnShowInfo()
{
return "myConsole";
}
#endregion
}
}
using System;
using System.Collections.Generic;
using System.Text;
using System.Windows.Forms;
using IMsg;namespace MYPlugin1{
public class MYDlg:Form,IMsgPlug
{
#region IMsgPlug 成員
public void OnShowDlg()
{
this.Text = "插件子窗體";
this.ShowDialog();
//調用Form的ShowDialog,顯示窗體
}
public string OnShowInfo()
{
return "MyDlg";
}
#endregion
}
}
將上面的都生成dll,生成目錄可以設置為新建exe工程的bin目錄plugins文件夾下。Plugins文件夾是新建的,專門存放插件的。 新建一個 WinForm項目,來使用剛才的插件.
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;using System.Text;
using System.Windows.Forms;using System.Collections;
using System.IO;
using System.Reflection;namespace MsgBoxMain{
public partial class FormMain : Form {
/// <summary>
/// 存放插件的集合
/// </summary>
private ArrayList plugins = new ArrayList();
public FormMain()
{
InitializeComponent();
}
/// <summary>
/// 載入所有插件
/// </summary>
private void LoadAllPlugs()
{
//獲取插件目錄(plugins)下所有文件
string[] files = Directory.GetFiles(Application.StartupPath + @"plugsins");
foreach (string file in files)
{
if (file.ToUpper().EndsWith(".DLL"))
{
try
{
//載入dll
Assembly ab = Assembly.LoadFrom(file);