结构型模式:Adapter 适配器模式

                                                          结构型模式:Adapter 适配器模式

1、适配(转换)的概念无处不在......
      适配,即在不改变原有实现的基础上,将原先不兼容的接口转换为兼容的接口。
  1)例1:现实中会将两箱插头转换为三箱插头;
  2)例2:现实中将220伏电压转换为想要的电压,供小电器使用流电;
  3)例3:现实中将水笼头接到较粗的水管上。
  以上三例中,三箱插头、电源适配器、水管接口都是适配器。
 
2、动机(Motivation)
  1)在软件系统中,由于应用环境的变化,常常需要将“一些现存的对象”放在新的环境中应用,但是新环境要求的接口是这些现存对象所不满足的。
  2)如何应对这种“迁移的变化”?如何既能利用现有对象的良 好实现,同时又能满足新的应用环境所要的接口?

3、意图(Intent)
      将一个类的接口转换成客户希望的另一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。
                          ——《设计模式》GoF
                         
4、实例:将ArrayList实现栈,先进后出
using system;
using System.Collections;

//栈的接口
public interface IStack //客户期望的接口
{
  void Push(object item); //将数据加入到栈
  void Pop();            //移除顶层的对象
  object Peek();  //只取顶层的对像
}

//这是一个对象适配器
public class MyStack : IStack //适配对像
{
  ArrayList list; //被适配的对像
  //ArrayList list2; //有可能有多个被适配的对象
  //构造器
  public MyStack()
  {
    list = new ArrayList();
  }
 
  public void Push(object item)
  {
    list.Add(item);
  }
  public void Pop()
  {
    list.RemoveAt(list.count - 1);
  }
  public object Peek()
  {
    return list[list.count - 1];
  }
}

//这是一个类适配器,不推荐使用类适配器,它违反了类的单一职责原则
public class Adapter : ArrayList, IStack //适配对像
{
  public void Push(object item)
  {
    this.Add(item);
  }
  public void Pop()
  {
    this.RemoveAt(list.count - 1);
  }
  public object Peek()
  {
    return this[list.count - 1];
  }
}

5、Adapter模式的几个要点
  1)Adapter模式主要应用于“希望复用一些现存的类,但是接口又是与复用环境要求不一致的情况”,在遗留代码复用、类库迁移等方面非常有用。
  2)GoF23种定了两种Adapter模式的实现结构:对象适配器和类适配器。但类适配器采用“多继承”的实现方式,带来了不良的高耦合,所以一般不推荐使用。对象适配器采用“对象组合”的方式,更符合松耦合精神。
  3)Adapter模式可以实现的非常灵活,不必拘泥于GoF23中定义的两种结构。
  例如:完全可以将Adapter模式中的“现存对象”作为新的接口方法参数,来达到适配的目的。
  4)Adapter模式本身要求我们尽可能地使用“面向接口的编程”风格,这样才能在后期很方便地适配。
      用例子来说明:‘尽可能地使用“面向接口的编程”风格’
//现有类
public class ExistingClass
{
  public void SpecificRequest1()
  {
  }
  public void SpecificRequest2()
  {
  }
}

//现在有一个新的环境需要复用ExistingClass
//新环境所使用的接口,后期将能很方便地适配。
public interface ITarget
{
  void Request();
}

//适配类
public class Adapter : ITarget
{
  ExistingClass adapter;
 
  public void Request()
  {
    adapter.SpecificRequest1();
    adapter.SpecificRequest2();
  }
}

//另一个系统
public class MySystem
{
  public void Process(ITarget target)
  {
    target.Request();
  }
}
 
6、.NET框架中的Adapter应用
  1)在.NET中复用COM对象:
    A:COM对象不符合.NET的接口
    B:使用tlbimp.exe来创建一个Runtime Callable Wrapper(RCW)以使其符合.NET对象的接口。
  2).NET数据访问类(Adapter变体)
    A:各种数据库并没有提供DataSet接口
    B:使用DbDataAdapter可以将任何各数据库访问/存取适配到一个DataSet对象上。
    C:用例子说明,如下:
public class EmployeeDAO
{
  public DataSet GetAllEmployee()
  {
    DbDataAdapter dataAdapter = new SqlDataAdapter();
    //有可能是连接Oracle
    //DbDataAdapter dataAdapter = new OracleDataAdapter();
    AdapterDataSet myDataSet = new AdapterDataSet();
    return myDataSet.GetDataSet(dataAdapter);
  }
}
//将任何各数据库访问/存取适配到一个DataSet对象上
class AdapterDataSet
{
  //DbDataAdapter是抽象类,更符合面向对象
  public DataSet GetDataSet(DbDataAdapter dataAdapter)
  {
    DataSet dataSet = new DataSet();
    dataAdapter.Fill(dataSet);
    return dataSet;
  }
}

  3)集合类中对现有对象的排序(Adapter变体)
    A:现有对象未实现IComparable接口
    B:实现一个排序适配器(继承IComparer接口),然后在其Compare方法中对两个对象进行比较。
    C:用例子说明,如下:
public class Employee
{
  int _age;
  string _name;
 
  public int Age
  {
    get{return this._age;}
    set{this._age = value;}
  }
 
  public string Name
  {
    get{return this._name;}
    set{this._name = value;}
  }
}

//排序适配
public class EmployeeSortAdapter : IComparer
{
  public int Compare(object obj1, object obj2)
  {
    if (obj1.GetType() != typeof(Employee) || obj2.GetType() != typeof(Employee))
    {
      //可以自定义一个异常
      throw new Exception();
    }
   
    Employee e1 = (Employee)obj1;
    Employee e2 = (Employee)obj2;
    //这里根据Age排序,当然也可以根据Name排序
    if (e1.Age == e2.Age)
    {
      return 0;
    }
    else if (e1.Age > e2.Age)
    {
      return 1;
    }
    else
    {
      return -1;
    }
  }
}

public class App
{
  public static void Main()
  {
    Employee[] employees = new Employee[100];
    Array.Sort(employees, new EmployeeSortAdapter());
  }
}

转载于:https://my.oschina.net/anyway/blog/15951

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值