动态加载类的原理——元数据的使用 (转载)

    技术2022-05-11  75

    在使用.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的信息以“元数据”的形式封装在程序或是组件中,又提供了一系列可以获取“元数据”的方法,所以我们可以程序运行期间来动态的访问这些信息。         具体的应用例子可以参看我的另一篇《动态加载类》。  参考资料:《C# Primer》和msdn相关文档。

    Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=256728

    我们在编写程序的时候经常会遇到这样的情况:程序中要用到某种计算,而且这种计算的计算方式很多,我们不得不在编写程序时就要考虑的十分全面,将各种情况到考虑到。但是这样做又非常的费力,因为我们无法预测到程序编好后,还会出现什么样的计算方式。如果计算方式是在交付给客户后,客户新提出的我们就不得不将新的计算方式写人程序中,然后重新编译,再交给客户。这样做是相当麻烦的,而且只为了这么一小段程序,就要重新编译整个工程,似乎代价也挺大的。    使用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();   }


    最新回复(0)