在使用.NET创建的程序或组件时,元数据(metadata)和代码(code)都存储于“自成一体”的单元中,这个单元称为装配件。我们可以在程序运行期间访问这些信息。
在System.Reflection中有这样一个class————Assembly,我们可以通过它来加载一个装配件。方法如下:
Assembly assm=Assembly.LoadFrom(fileName);
其中filename是要加载的装配件的文件名称(带路径)。
接下来,我们就可以通过使用System.Reflection内提供的Info classes来获取装配件中的信息了。首先让我们看一下这些Info classes:
MethodInfo 获取某个“成员函数”的信息,并提供对此“成员函数”元数据的访问。
ParameterInfo 获取某个“参数”的信息,并提供对此“参数”元数据的访问。
Constructorinfo 获取某个“构造函数”的信息,并提供对此“构造函数”元数据的访问。
PropertyInfo 获取某个“属性”的信息,并提供对此“属性”元数据的访问。
FieldInfo 获取某个“数据成员”的信息,并提供对此“数据成员”元数据的访问。
EventInfo 获取某个“事件”的信息,并提供对此“事件”元数据的访问。
上面列出的这些classes(除ParameterInfo外)的访问操作,要通过一个Type对象来完成。比如我们要获得一个装配件的“成员函数”就要这样做:
System.Reflection.Assembly ass=System.Reflection.Assembly.LoadFrom(fileName);
Type[] tp=ass.GetTypes();
System.Reflection.MethodInfo[] mi=tp[0].GetMethods();
使用同样的方法我们还可以得到其它的信息,如下:
获得“构造函数”信息:System.Reflection.ConstructorInfo[] ci=tp[0].GetConstructors();
获得“属性”信息:System.Reflection.PropertyInfo[] pi=tp[0].GetProperties();
获得“数据成员”信息:System.Reflection.FieldInfo[] fi=tp[0].GetFields();
获得“事件”信息:System.Reflection.EventInfo[] ei=tp[0].GetEvents();
此外,我们可以通过ParameterInfo类来获取“成员函数”和“构造函数”的参数信息,如下:
获取“成员函数”的参数信息:System.Reflection.ParameterInfo[] pi=mi[0].GetParameters();
获取“构造函数”的参数信息:System.Reflection.ParameterInfo[] pi=ci[0].GetParameters();
ParameterInfo类有两个重要的属性:Name和ParameterType。通过它们我们可以得到“参数”的名称和数据类型。
由于.NET将class的信息以“元数据”的形式封装在程序或是组件中,又提供了一系列可以获取“元数据”的方法,所以我们可以程序运行期间来动态的访问这些信息。
我们在编写程序的时候经常会遇到这样的情况:程序中要用到某种计算,而且这种计算的计算方式很多,我们不得不在编写程序时就要考虑的十分全面,将各 种情况到考虑到。但是这样做又非常的费力,因为我们无法预测到程序编好后,还会出现什么样的计算方式。如果计算方式是在交付给客户后,客户新提出的我们就 不得不将新的计算方式写人程序中,然后重新编译,再交给客户。这样做是相当麻烦的,而且只为了这么一小段程序,就要重新编译整个工程,似乎代价也挺大的。
使用MS.NET中System.Reflection中的一些方法,可以帮助我们很好的解决上面的问题。 首先,在遇到上面提出的问题的时候,我们先要进行一下分析,这种计算需要一些什么参数?在不同的计算方式中,它们共同的参数是什么?不同的计算方式中特有的参数是否可以通过共有的参数计算出来,或是通过其它方法获得?分析完后,提取出可用的共同参数。
接下来,我们就可以编写计算方法了。将这种计算的每一种方式都写成一个DLL一个类中的方法,并将其编译为一个DLL文件。MS.NET中,类的格式要定 死,也就是说编写的类的namespace和class要一样,类中的方法名称也必须是一样的。而且,方法的参数就是上面所说的共同参数。
然后,将编译好的DLL文件放在同一文件夹内,随程序一起发布就可以了。
在程序中可以这样处理所要用到的不同计算方式:给每一种计算方式起一个名字(客户能够明白的),然后将这些名字放在下拉列表框的text属性中,并将对应 的DLL文件名放在下拉列表框的value属性中。这样,用户选择不同的计算方式就可以调用不同DLL文件中的计算方法了。
下面是一个简单的示例:
我将计算方式的名字和DLL文件名放在一个xml文件中,程序加载时将它们读取到下拉列表框中。方法如下:(asp.net中)
// 在此处放置用户代码以初始化页面
if(Page.IsPostBack==false)
{
// 页面首次加载时读取XML文件
System.Xml.XmlDocument xmlDoc=new System.Xml.XmlDocument();
System.Xml.XmlNode xmlNd;
int i;
xmlDoc.Load(Server.MapPath("MyConfig.xml"));
xmlNd=xmlDoc.SelectSingleNode("//dllfile");
// 将读出的XML文件的有关内容写入下拉列表框中
for(i=0;i<xmlNd.SelectNodes("field").Count;i++)
{
ListItem it=new ListItem();
it.Text=xmlNd.SelectNodes("field").Item(i).Attributes["text"].Value;
it.Value=xmlNd.SelectNodes("field").Item(i).Attributes["filename"].Value;
this.ddlType.Items.Add(it);
}
}
然后,在输入完参数之后,可以使用下面的代码来完成计算并将结果显示出来。我写的方法是计算后返回一个DataTable。
DataTable dt=null;
// 加载类所在的dll文件
Assembly ass=Assembly.LoadFrom(Server.MapPath(this.ddlType.SelectedItem.Value));
// 获取类型
Type tp=ass.GetType("MyNamespace.MyClass");
// 获取方法
MethodInfo mi=tp.GetMethod("MyMethodl");
// 创建实例
Object obj=System.Activator.CreateInstance(tp);
// 付值参数数组(要求类型一致性)
Object[] objArray=new object[7];
objArray[0]=Convert.ToDouble(this.textBox1.Text);
objArray[1]=Convert.ToInt32(this.textBox2.Text);
objArray[2]=Convert.ToDouble(this.textBox3.Text);
// 调用方法
dt=(DataTable)mi.Invoke(obj,objArray);
// 有返回值就绑定到DataGrid
if(dt!=null)
{
this.grdResult.DataSource=dt;
this.grdResult.DataBind();
}