深入理解设计模式(18):适配器模式
一、什么是适配器模式
定义:适配器模式属于结构型模式,把一个类的接口变成客户端所期待的另一种接口,从而使原本接口不匹配而无法一起工作的两个类能够在一起工作。
适配器模式又可以分为4种类型,类适配器模式、对象适配器模式、单接口适配器模式(缺省适配器模式)和双向适配器模式。后2种模式的实现比较复杂并且在实际开发过程中很少使用。
二、适配器模式的结构
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就是非常好的手段,我们不必关注不同数据库的数据细节,就可以灵活的使用数据。
上一篇: web开发布局---传统布局篇
下一篇: 详解python校验SQL脚本命名规则