每個.net組件除了程式碼外都額外包含了元資料。元資料包括了程序集本身的信息,例如版本號,引用了什麼程序集,所有類型的信息,包括其方法、屬性、字段。使用.net反射,可以在運行時讀取這些訊息,並且可以動態地呼叫方法。
專案快完了,終於有時間寫blog了,,
做一個動態呼叫組件指定方法的範例。
項目1(Demo)中包含一個Test類別,Test類別中寫了一個getList方法,這個方法回傳的資料是手工加入的。原始碼如下:
項目1
using System;
using System.Collections.Generic;
using System.Text;
using System.Data;
namespace Demo
{
public class Test
{
public DataTable getList(string id)
{
DataTable dt = new DataTable();
dt.Columns.Add(new DataColumn("id"));
dt.Columns.Add(new DataColumn("name"));
dt.Columns.Add(new DataColumn("sex"));
DataRow dr = dt.NewRow();
dr["id"] = "zl";
dr["name"] = "張鈴";
dr["sex"] = "男";
dt.Rows.Add(dr);
dr = dt.NewRow();
dr["id"] = "zl";
dr["name"] = "李四";
dr["sex"] = "女";
dt.Rows.Add(dr);
return dt;
}
}
}
項目2(DemoXml)中包含一個Test類別,Test類別中寫了一個getList方法,這個方法回傳的資料是從資料庫讀取的。原始碼如下:
項目2
using System;
using System.Collections.Generic;
using System.Text;
using System.Data;
using System.Data.SqlClient;
using System.Xml;
namespace DemoXml
{
public class Test
{
private SqlConnection cn;
public DataTable getList(string id)
{
try
{
cn = new SqlConnection(System.Configuration.ConfigurationSettings.AppSettings["pubs"]);
SqlCommand cmd = new SqlCommand();
SqlDataAdapter da = new SqlDataAdapter();
cmd.CommandText = "SELECT au_id as id,au_lname as name,au_fname as sex from authors";
cmd.CommandType = CommandType.Text;
cmd.Connection = cn;
da.SelectCommand = cmd;
DataTable dt = new DataTable();
da.Fill(dt);
return dt;
}
catch (Exception ex)
{
throw new ApplicationException("出現例外狀況:"+ex.Message+ex.StackTrace);
}
finally
{
cn.Close();
cn = null;
}
}
}
}
項目3(WebDemo)中示範動態以指定程式集中getList的方法傳回一個DataTable,用一個gridview顯示其傳回的資料。
調用演示
using System;
using System.Data;
using System.Configuration;
using System.Web;
using System.Web.Security;
using System.Web.UI;
using System.Web.UI.WebControls;
using System.Web.UI.WebControls.WebParts;
using System.Web.UI.HtmlControls;
using System.Reflection;
public partial class _Default : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
if (!IsPostBack)
{
DropBind();
}
}
資料初始化,可設定在web.config檔中#region 資料初始化,可設定在web.config檔中
public void DropBind()
{
DataTable dt = new DataTable();
dt.Columns.Add(new DataColumn("name"));
dt.Columns.Add(new DataColumn("filepath"));
DataRow dr = dt.NewRow();
dr["name"] = "載入自己定義資料";
dr["filepath"] = Server.MapPath(@"FilesDemo.dll");
dt.Rows.Add(dr);
dr = dt.NewRow();
dr["name"] = "載入xml資料";
dr["filepath"] = Server.MapPath(@"FilesDemoXml.dll");
dt.Rows.Add(dr);
this.DropDownList1.DataSource = dt;
this.DropDownList1.DataTextField = "name";
this.DropDownList1.DataValueField = "filepath";
this.DropDownList1.DataBind();
}
#endregion
protected void DropDownList1_SelectedIndexChanged(object sender, EventArgs e)
{
try
{
//讀取選擇指定的dll文件
string strPath = (sender as DropDownList).SelectedValue.Trim();
string NameSpace = this.DropDownList1.SelectedIndex == 0 ? "Demo.Test" : "DemoXml.Test";
//載入指定的組件記憶體中
Assembly assembly = Assembly.LoadFrom(strPath);
//返加程式集中的一個指定的對象,哪果是傳回所有對象,則用GetTypes()傳回一個Typt物件的陣列.
Type T = assembly.GetType(NameSpace);
//返回方法資訊(公共方法)
MethodInfo mi = T.GetMethod("getList");
//根據前面type類型建立一個對象
object o = Activator.CreateInstance(T);
//參數
object[] par = new object[] { "E01" };
//透過MethodInfo物件的Invoke方法,動態呼叫此方法,參數o是因為實例方法需要在呼叫時有一個實例存在
DataTable dt = (DataTable)mi.Invoke(o, par);
this.GridView1.DataSource = dt;
this.GridView1.DataBind();
}
catch (Exception ex)
{
//do Exception
}
}
}
透過Assembly.LoadFrom方法傳回的Assembly對象,可以讀取其中的元資料。其中的GetType會傳回一個用來表示指定組件的type物件(讀取程式集的所有型別用GetTypes會傳回一個type物件的陣列)。
返回方法資訊(公共方法)
MethodInfo mi = T.GetMethod("getList");
根據前面type類型建立一個對象
object o = Activator.CreateInstance(T);
參數
object[] par = new object[] { "E01" };
透過MethodInfo物件的Invoke方法,動態呼叫此方法,參數o是因為實例方法需要在呼叫時有一個實例存在.
DataTable dt = (DataTable)mi.Invoke(o, par);
呼叫傳回的資料顯示清單中。
範例下載: