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

 
        在使用.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();
   }

阅读更多
个人分类: .Net
上一篇有关动态控件的问题
下一篇手足球有感
想对作者说点什么? 我来说一句

没有更多推荐了,返回首页

关闭
关闭
关闭