欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

深入理解设计模式(18):适配器模式

程序员文章站 2022-05-25 18:42:51
一、什么是适配器模式 定义:适配器模式属于结构型模式,把一个类的接口变成客户端所期待的另一种接口,从而使原本接口不匹配而无法一起工作的两个类能够在一起工作。 适配器模式又可以分为4种类型,类适配器模式、对象适配器模式、单接口适配器模式(缺省适配器模式)和双向适配器模式。后2种模式的实现比较复杂并且在 ......

一、什么是适配器模式

定义:适配器模式属于结构型模式,把一个类的接口变成客户端所期待的另一种接口,从而使原本接口不匹配而无法一起工作的两个类能够在一起工作。

适配器模式又可以分为4种类型类适配器模式、对象适配器模式、单接口适配器模式(缺省适配器模式)和双向适配器模式。后2种模式的实现比较复杂并且在实际开发过程中很少使用。

二、适配器模式的结构

深入理解设计模式(18):适配器模式

 

adaptee:初始角色,实现了我们想要使用的功能,但是接口不匹配

target:目标角色,定义了客户端期望的接口

adapter:适配器角色,实现了目标接口。实现目标接口的方法是:内部包含一个adaptee的对象,通过这个对象调用adaptee的原有方法实现目标接口。(注:这里说的是对象适配器)

三、适配器模式的使用场景

  当前打开我这篇文章的笔记本电脑,电源的另一边不正连着一块适配器吗?你平时想将三口插座插进二口插座里面,不也需要一个适配器吗?整天插在插座上的手机充电头,不也是一个适配器吗?

  1、系统需要复用现有类,而该类的接口不符合系统的需求;
  2、想要建立一个可重复使用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作;
  3、对于对象适配器模式,在设计里需要改变多个已有子类的接口,如果使用类的适配器模式,就要针对每一个子类做一个适配器,而这不太实际。

四、适配器模式的优缺点

优点:

  1、可以让任何两个没有关联的类一起运行;
  2、可以在不修改原有代码的基础上来复用现有类,很好地符合 “开闭原则”;
  3、增加了类的透明度和更好的灵活性。

缺点:

  1、由于c#不支持多重继承,所以最多只能适配一个适配者类,而且目标类必须是抽象类;
  2、采用了类和接口的“双继承”实现方式,带来了不良的高耦合。

五、适配器模式的实现

 

1.类适配器模式

 

namespace 设计模式之适配器模式
{
    /// <summary>
    /// 这里手机充电器为例,我们的家的插座是两相电的,但是手机的插座接头是三相电的
    /// </summary>
    class client
    {
        static void main(string[] args)
        {
            //好了,现在可以充电了
            itwoholetarget change = new threetotwoadapter();
            change.request();
            console.readline();
        }
    }
 
    /// <summary>
    /// 我家只有2个孔的插座,也就是适配器模式中的目标角色(target),这里只能是接口,也是类适配器的限制
    /// </summary>
    public interface itwoholetarget
    {
        void request();
    }
 
    /// <summary>
    /// 3个孔的插头,源角色——需要适配的类(adaptee)
    /// </summary>
    public abstract class threeholeadaptee
    {
        public void specificrequest()
        {
            console.writeline("我是三个孔的插头");
        }
    }
 
    /// <summary>
    /// 适配器类,接口要放在类的后面,在此无法适配更多的对象,这是类适配器的不足
    /// </summary>
    public class threetotwoadapter:threeholeadaptee,itwoholetarget
    {
        /// <summary>
        /// 实现2个孔插头接口方法
        /// </summary>
        public void request()
        {
            // 调用3个孔插头方法
            this.specificrequest();
        }
    }
}

 

2.对象适配器模式

 

namespace 对象的适配器模式
{
    ///<summary>
    ///家里只有两个孔的插座,也懒得买插线板了,还要花钱,但是我的手机是一个有3个小柱子的插头,明显直接搞不定,那就适配吧
    ///</summary>
    class client
    {
        static void main(string[] args)
        {
            //好了,现在就可以给手机充电了
            twoholetarget hometwohole = new threetotwoadapter();
            hometwohole.request();
            console.readline();
        }
    }

    /// <summary>
    /// 我家只有2个孔的插座,也就是适配器模式中的目标(target)角色,这里可以写成抽象类或者接口
    /// </summary>
    public class twoholetarget
    {
        // 客户端需要的方法
        public virtual void request()
        {
            console.writeline("两孔的充电器可以使用");
        }
    }

    /// <summary>
    /// 手机充电器是有3个柱子的插头,源角色——需要适配的类(adaptee)
    /// </summary>
    public class threeholeadaptee
    {
        public void specificrequest()
        {
            console.writeline("我是3个孔的插头也可以使用了");
        }
    }

    /// <summary>
    /// 适配器类,twohole这个对象写成接口或者抽象类更好,面向接口编程嘛
    /// </summary>
    public class threetotwoadapter : twoholetarget
    {
        // 引用两个孔插头的实例,从而将客户端与twohole联系起来
        private threeholeadaptee threeholeadaptee = new threeholeadaptee();
        //这里可以继续增加适配的对象。。

        /// <summary>
        /// 实现2个孔插头接口方法
        /// </summary>
        public override void request()
        {
            //可以做具体的转换工作
            threeholeadaptee.specificrequest();
            //可以做具体的转换工作
        }
    }
}

六、适配器模式的.net应用

  在.net中有一个类库已经实现且非常重要的适配器,那就是dataadapter。dataadapter用作dataset和数据源之间的适配器以便检索和保存数据,dataadapter通过映射fill(这更改了dataset中的数据以便与数据源中的数据相匹配)和update(这更改了数据源中的数据以便与dataset中的数据相匹配)来提供这一适配器。

  由于数据源可能来自于sql server,可能来自于oracel,也可能来自于db2、mysql,这些数据在组织上可能有不同之处,但我们希望得到统一的dataset(实质是xml数据),此时用dataadapter就是非常好的手段,我们不必关注不同数据库的数据细节,就可以灵活的使用数据。