Pluge模式?反射?后期绑定?

我不知道有没有这个设计模式,在网上搜索过也没有找到多少,只找到两篇文章。
Pluge模式的功能是后期绑定还是条件外置,希望有知道的朋友确定一下。
举例:
一.通过配置文件:条件外置与反射实例
接口:
namespace SamplePlugePattern.Interface
{
    /// <summary>
    /// define a action interface
    /// </summary>
    public interface IAction
    {
        void Burn();
    }
}

实现:
namespace Machine
{
    public class MachineT50:IAction
    {
        public void Burn()
        {
            Console.WriteLine("MachineT50 is burning now.");
        }
    }
}

客户类:
using System;
using System.Collections.Generic;
using System.Reflection;
using System.Configuration;
using SamplePlugePattern.Interface;

namespace WorkPlat
{
    /// <summary>
    /// Pluge Pattern Sample
    /// </summary>
    /// <remarks>author PetterLiu http://wintersun.cnblogs.com </remarks>
    class Program
    {
        static void Main(string[] args)
        {
            string AssemblyName = ConfigurationManager.AppSettings["AssemblyName"];
            string TypeName = ConfigurationManager.AppSettings["TypeName"];
            IAction action = UsingActivator(AssemblyName, TypeName);
            action.Burn();
        }

        private static IAction UsingCurrentAppDomain(string assemblyname, string typename)
        {
            return AppDomain.CurrentDomain.CreateInstanceAndUnwrap(assemblyname, typename) as IAction;
        }

        private static IAction UsingActivator(string assemblyname, string typename)
        {
            Assembly assembly = Assembly.Load(assemblyname);
            Type type = assembly.GetType(typename);
            return Activator.CreateInstance(type) as IAction;
        }
    }
}

这里使用的是appSettings,实际中还可以用自定义配制节。
<configuration>
  <appSettings>
    <add key="AssemblyName" value="Machine"/>
    <add key="TypeName" value="Machine.MachineT50"/>
  </appSettings>
</configuration>

总结:为了使用一个在编译期间未知的类,在编译时未知的类型实现一个接口,而该接口是编译器所知的。为此,我们不得不创建第三个程序集:
第一个程序集:定义一个接口;
第二个程序集:是要被反射的程序集(要引用第一个程序集,并且要实现第一个程序集中定义的接口);
第三个程序集:使用反射的程序集(也要引用第一个程序集,这样可以使用第一个程序集的接口)。
二.C#后期绑定方式来调用COM对象
实际应用当中,可能有的时候需要用到后期绑定方式来调用COM对象。

1、静态方法System.Type.GetTypeFromProgID 方法 (String),该方法可获取与指定程序标识符 (ProgID) 关联的类型,如果在加载 Type 时遇到错误,则返回空值。
通过它可获取对代表COM对象类型的Type对象的引用。此方法是专为 COM 组件支持提供的。
2、静态方法System.Activator.CreateInstance(),使用与指定参数匹配程度最高的构造函数创建指定类型的实例。
这里使用Activator.CreateInstance()创建COM对象的一个实例。
3、System.Type类的非静态方法InvokeMember()可以创建一个在编译期间未知的类的实例,只需要在调用的时候使用BindingFlags枚举量中的SetProperty值即可对com对象的公共属性进行操作。
InvokeMember方法的5个参数如下:  
  1)要调用的成员的名称。String,它包含要调用的构造函数、方法、属性或字段成员的名称;空字符串 (""),表示调用默认成员;对于 IDispatch 成员,则为一个表示 DispID 的字符串,例如“[DispID=3]”。
  2)BindingFlag枚举的值,表明调用的是属性还是方法等。这个位屏蔽,由一个或多个指定搜索执行方式的 BindingFlags 组成。访问可以是 BindingFlags 之一,如 Public、NonPublic、Private、InvokeMethod 和 GetField 等。不需要指定查找类型。如果省略查找类型,则将应用 BindingFlags.Public |BindingFlags.Instance。
  3)一个 Binder 对象,该对象定义一组属性并启用绑定,而绑定可能涉及选择重载方法、强制参数类型和通过反射调用成员。通常使用默认Binder 对象,即传递一个null值;
  4)对Com对象本身的引用;
  5)包含传递给要调用的成员的参数的数组。即希望给Com方法发送的输入参数数组。

以下是测试代码,调用Word的com组件,并设置它的可视属性。

           object comObject;
            System.Type comObjectName;
            //参数
            object[] args = new object[1];
            //要获取的类型的 ProgID。
            string progID = "Word.Application";

            //与指定 ProgID 关联的类型,即获取相应的Com对象
            comObjectName = System.Type.GetTypeFromProgID(progID);
            //创建Com的实例
            if (comObjectName != null)
            {
                comObject = Activator.CreateInstance(comObjectName);
                //设置需要设置的参数值
                args[0] = true;
                //设置可视属性,显示Word窗体
                comObjectName.InvokeMember("Visible", BindingFlags.SetProperty, null, comObject, args);
            }
以下是调用方法示例(关闭刚才创建的Word窗体)             args = new object[3];
            comObjectName.InvokeMember("Quit", BindingFlags.InvokeMethod, null, comObject, args);

posted on 2009-11-03 22:57  AngelLee2009  阅读(240)  评论(0编辑  收藏  举报

导航