【JAVA今法修真】 第九章 兼容并包 适配模式

551 阅读20分钟

今法修真.jpg

您好,我是南橘,万法仙门的掌门,刚刚从九州世界穿越到地球,因为时空乱流的影响导致我的法力全失,现在不得不通过这个平台向广大修真天才们借去力量。你们的每一个点赞,每一个关注都是让我回到九州世界的助力,兄弟萌来为我注入修为吧!关注WX号:南橘ryc

等我回去以后,大家都是万法仙门的长老,我会给大家数不尽的天材地宝,人人如龙,全民飞升。

万维仙网正如它的名字一样,方法在一个空间内叠加了无数维度,利用超链接技术,可以自由的在各个维度内穿梭。

李长庚刚刚进入的时候出现在一个模仿万法仙门山门的地方,周边漂浮着各式各样的彩色气泡,气泡折射着里面小世界的剪影,身边偶尔有其他弟子出现,又匆匆忙忙的通过小气泡前往其他空间。

“看师弟在这里站了那么久,不像是卡了,再看你这身打扮,应该是新来的吧?”一个穿着十分京剧风的小女孩停在李小庚的面前。

“是的,师姐有什么事情吗?”李小庚笑着回答,与这里匆匆而过的人相比,他因为穿着普通而和大家格格不入。

对方笑了笑,向李小庚推送过一个数据包:“也没什么事情,师弟你第一次进入万法内网,所以不认识我。我是万法仙门这届的首席弟子周明瑞,不过一般大家都叫我克莱恩。这个数据包里记载了一些万法仙门内网的参考资料和搜索规则,还有一些新手装饰,师弟可以自行改变自己的外貌。”

“啊!克总你怎么打扮的跟云堇似的?”李小庚被狠狠地震撼了一下。

“一看师弟你就很少上网。”克总又捂嘴笑:“师弟,我只是周明瑞本人留在这里接待大家的人工智能而已,至于为什么打扮成云堇的样子,不过是这段时间的xp而已。”

李小庚又和披着云堇皮肤的周明瑞聊了一段时间,不多久,又有新人进入内网,李小庚便不再停留,按照周明瑞给的指引前往下一个空间。

这次的目的地,万法设计中心。

万法设计中心是整个九州最大的法器设计中心之一,通过23种设计模式的组合与优化,无数有着各式各样特点的法器诞生于此,甚至炼器宗常年有上百个工作组在此交流,互相取长补短。

当然,在万法内网上,设计中心主要承担的功能还是模拟与教学。

“你想明白生命的意义吗?你想真正的活着吗?那就来购买最新款的先天卜算仪吧!”

“作为一个普通人,你难道不羡慕天道剑宗的剑修们御剑乘风去,神游天地间的样子吗。礼貌阁新款外接金丹,筑基期即可使用,最大输出功率高达12mb,内附神剑御雷真诀与剑二十三,圆你一个做剑仙的梦。”

刚刚进入设计中心的区域,扑面而来的广告流瞬间出现在李小庚的眼前,艰难的无视了这些看起来十分诱人的信息后,李小庚按照周明瑞给的搜索规则找到了他想看的东西。

设计模式

一、设计模式的七条原则

1、开闭原则

开闭原则的核心是对扩展开放,对修改封闭。在程序需要进行拓展的时候,不能去修改原有的代码,而是要扩展原有代码,实现一个热插拔的效果。所以一句话概括就是:为了使程序的扩展性好,易于维护和升级。

可以通过“抽象约束、封装变化”来实现开闭原则,即通过接口或者抽象类为软件实体定义一个相对稳定的抽象层,而将相同的可变因素封装在相同的具体实现类中。

2、单一职责原则

单一职责原则规定一个类应该有且仅有一个引起它变化的原因,不要存在多于一个导致类变更的原因,否则就应该把类拆分。

单一职责原则是最简单但又最难运用的原则,需要设计人员发现类的不同职责并将其分离,再封装到不同的类或模块中。

3、里氏替换原则

里氏替换原则主要阐述了有关继承的一些原则,也就是什么时候应该使用继承,什么时候不应该使用继承,以及其中蕴含的原理。里氏替换原是继承复用的基础,它反映了基类与子类之间的关系,是对开闭原则的补充,是对实现抽象化的具体步骤的规范。

里氏替换原则通俗来讲就是:子类可以扩展父类的功能,但不能改变父类原有的功能。

4、依赖倒转原则

依赖倒置原则的目的是通过要面向接口的编程来降低类间的耦合性,它是实现开闭原则的重要途径之一,它降低了客户与实现模块之间的耦合。其核心思想是:要面向接口编程,不要面向实现编程。

4、接口隔离原则

接口隔离原则要求我们尽量将臃肿庞大的接口拆分成更小的和更具体的接口,让接口中只包含客户感兴趣的方法。 每个接口中不存在子类用不到却必须实现的方法,如果不然,就要将接口拆分。使用多个隔离的接口,比使用单个接口要好。

5、迪米特法则

迪米特法则的定义是一个类对自己依赖的类知道的越少越好。无论被依赖的类多么复杂,都应该将逻辑封装在方法的内部,通过public方法提供给外部。这样当被依赖的类变化时,才能最小的影响该类。

迪米特法则的另一个表达方式是:只与直接的朋友通信。类之间只要有耦合关系,就叫朋友关系。耦合分为依赖、关联、聚合、组合等。我们称出现为成员变量、方法参数、方法返回值中的类为直接朋友。局部变量、临时变量则不是直接的朋友。我们要求陌生的类不要作为局部变量出现在类中。

6、合成复用原则

合成复用原则又叫组合/聚合复用原则,它要求在软件复用时,要尽量先使用组合或者聚合等关联关系来实现,其次才考虑使用继承关系来实现。

如果要使用继承关系,则必须严格遵循里氏替换原则。合成复用原则同里氏替换原则相辅相成的,两者都是开闭原则的具体实现规范。

“师弟是第一次来设计中心吧?”一个亚古兽出现在了李小庚的身边,作揖道:“我是设计中心的设计师伍彦光,轮班作为内网的新人指导员,你有什么不懂的可以问我。”

万法仙门不愧是万法仙门,进了内网后很难看见与二次元无关的生物。

看到对方如此客气,李小庚也连忙回礼:“伍师兄你好,我想来设计中心学习一下设计模式。”

“很好啊,如果是来学习的话,不知道之前有无基础?”

“略懂一些。”

“你可知设计模式如何分类的?”

“今法中通常根据模式是用来完成什么工作来划分,这种方式可分为创建型模式结构型模式行为型模式,根据运用的范围来划分则式可分为类模式对象模式两种**

二、设计模式的划分

设计模式的本质是面向对象设计原则的实际运用,是对类的封装性、继承性和多态性以及类的关联关系和组合关系的充分理解

创建型模式的主要关注点是“怎样创建对象”,它的主要特点是将对象的创建与使用分离。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节,对象的创建由相关的工厂来完成。就像我们去商场购买商品时,不需要知道商品是怎么生产出来一样,因为它们由专门的厂商生产。

结构型模式主要描述如何将类或对象通过组合来实现功能。它主要分为类结构型模式对象结构型模式类结构型模式采用继承机制来组织接口和类,对象结构型模式釆用组合或聚合来组合对象。

行为型模式是对在不同的对象之间划分责任算法的抽象化,用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象无法单独完成的任务,它涉及算法与对象间职责的分配。

类模式:用于处理类与子类之间的关系,这些关系通过继承来建立,是静态的,在编译时刻便确定下来了

对象模式:用于处理对象之间的关系,这些关系可以通过组合或聚合来实现,在运行时刻是可以变化的,更具动态性

范围\目的创建型模式结构型模式行为型模式
类模式工厂方法(类)适配器模板方法、解释器
对象模式单例 原型 抽象工厂 建造者代理 (对象)适配器 桥接 装饰 外观 享元 组合策略 命令 职责链 状态 观察者 中介者 迭代器 访问者 备忘录

创建型模式分为以下5种:

  • 单例(Singleton)模式:某个类只能生成一个实例,该类提供了一个全局访问点供外部获取该实例,其拓展是有限多例模式。
  • ** 原型(Prototype)模式**:将一个对象作为原型,通过对其进行复制而克隆出多个和原型类似的新实例。
  • 工厂方法(FactoryMethod)模式:定义一个用于创建产品的接口,由子类决定生产什么产品。
  • 抽象工厂(AbstractFactory)模式:提供一个创建产品族的接口,其每个子类可以生产一系列相关的产品。
  • 建造者(Builder)模式:将一个复杂对象分解成多个相对简单的部分,然后根据不同需要分别创建它们,最后构建成该复杂对象。

结构型模式分为以下 7 种:

  • 代理(Proxy)模式:为某对象提供一种代理以控制对该对象的访问。即客户端通过代理间接地访问该对象,从而限制、增强或修改该对象的一些特性。
  • 适配器(Adapter)模式:将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。
  • 桥接(Bridge)模式:将抽象与实现分离,使它们可以独立变化。它是用组合关系代替继承关系来实现的,从而降低了抽象和实现这两个可变维度的耦合度。
  • 装饰(Decorator)模式:动态地给对象增加一些职责,即增加其额外的功能。
  • 外观(Facade)模式:为多个复杂的子系统提供一个一致的接口,使这些子系统更加容易被访问。
  • 享元(Flyweight)模式:运用共享技术来有效地支持大量细粒度对象的复用。
  • 组合(Composite)模式:将对象组合成树状层次结构,使用户对单个对象和组合对象具有一致的访问性。

行为型模式是设计模式中最为庞大的一类,它包含以下 11 种模式:

  • 模板方法(Template Method)模式:定义一个操作中的算法骨架,将算法的一些步骤延迟到子类中,使得子类在可以不改变该算法结构的情况下重定义该算法的某些特定步骤。
  • 策略(Strategy)模式:定义了一系列算法,并将每个算法封装起来,使它们可以相互替换,且算法的改变不会影响使用算法的客户。
  • 命令(Command)模式:将一个请求封装为一个对象,使发出请求的责任和执行请求的责任分割开。
  • 职责链(Chain of Responsibility)模式:把请求从链中的一个对象传到下一个对象,直到请求被响应为止。通过这种方式去除对象之间的耦合。
  • 状态(State)模式:允许一个对象在其内部状态发生改变时改变其行为能力。
  • 观察者(Observer)模式:多个对象间存在一对多关系,当一个对象发生改变时,把这种改变通知给其他多个对象,从而影响其他对象的行为。
  • 中介者(Mediator)模式:定义一个中介对象来简化原有对象之间的交互关系,降低系统中对象间的耦合度,使原有对象之间不必相互了解。
  • 迭代器(Iterator)模式:提供一种方法来顺序访问聚合对象中的一系列数据,而不暴露聚合对象的内部表示。
  • 访问者(Visitor)模式:在不改变集合元素的前提下,为一个集合中的每个元素提供多种访问方式,即每个元素有多个访问者对象访问。
  • 备忘录(Memento)模式:在不破坏封装性的前提下,获取并保存一个对象的内部状态,以便以后恢复它。
  • 解释器(Interpreter)模式:提供如何定义语言的文法,以及对语言句子的解释方法,即解释器。

“看来师弟是早有研究。”伍彦光笑道:“不知道今天来是想学什么?”

“实不相瞒,我是云霄殿的新人弟子李小庚,师兄你也知道,我们云霄殿兵微将寡,但是法宝什么的实在是太多,入门没几年,年筑基期都没过去,已经六神装了,没办法,因为境界低微,所以需要学习适配器模式来兼容一下不同属性的法宝。”

伍彦光嘴角微抽,太阳穴位置的三叉神经微微跳起,深呼吸了好几次才缓缓开口道:“好的师弟,我来给你讲一讲我对适配器模式的理解。”

三、适配器模式

适配器模式是将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。适配器模式分为类结构型模式和对象结构型模式两种,前者类之间的耦合度比后者高,且应用相对较少些。

适配器模式的优点:

1、客户端通过适配器可以透明地调用目标接口。
2、复用了现存的类,不需要修改原有代码而重用现有的适配者类。
3、将目标类和适配者类解耦,解决了目标类和适配者类接口不一致的问题。

适配器模式的缺点:

1、适配器编写过程需要结合业务场景全面考虑,可能会增加系统的复杂性。
2、增加代码阅读难度,降低代码可读性,过多使用适配器会使系统代码变得凌乱。

在实际的工作中,有时候会出现公司合并或者收购别的公司的情况(确信),这个时候,就会把其他公司的代码给接收过来,如果遇到相同相似的模块,往往会需要代码合并,适配器模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。或是在使用第三方提供的组件,但组件接口定义和自己要求的接口定义不同,使用适配器模式来解决这个问题。亦或是以前开发的系统存在满足新系统功能需求的类,但其接口同新系统的接口不一致

以上种种情况,都是我们使用适配器模式的理由。

再提一句:设计模式就是前人经验的结晶,每一种设计模式都解决了特定的问题,使用这些设计模式就是站在了前人的肩膀上。设计模式更多的是让我们学习前人的经验,而不是为了用而用,为了炫技而炫技。

适配器模式的主要角色:

目标接口:当前系统业务所期待的接口,它可以是抽象类或接口。
适配者类:它是被访问和适配的现存组件库中的组件接口。
适配器类:它是一个转换器,通过继承或引用适配者的对象,把适配者接口转换成目标接口,让客户按目标接口的格式访
问适配者。

四、类适配器模式实现

1、目标接口

public interface Smoke {
    //烟草专卖
    public void sellSmoke();
}

2、适配者类

public class GuangdongSmoke implements Smoke{
    //广东省售烟平台在系统内,可以卖烟
    public void companyGD(){
        System.out.println("广东省烟草专卖");
    }

    @Override
    public void sellSmoke() {
        this.companyGD();
    }
}

public class GuiZhouSmoke {
    //贵州省没有接入系统,无法卖烟
    public void companyGZ(){
        System.out.println("贵州省接入烟草专卖");
    }

}

3、适配器类

public class Adapter extends GuiZhouSmoke implements Smoke {

    /**
     *     适配器类,继承了被适配类,实现标准接口
     *     就相当于通过中间代码将贵州的烟草信息接入广州
      */
    @Override
    public void sellSmoke() {
        super.companyGZ();
    }
}

4、测试类

  public static void main(String[] args) {
        //本身功能实现
        Smoke gd =new GuangdongSmoke();
        gd.sellSmoke(); //广东省烟草专卖
        //通过适配器来实现
        Smoke gz =new Adapter();
        gz.sellSmoke();//贵州省接入烟草专卖
    }

上面这种实现的适配器称为类适配器,因为 Adapter 类既继承了 GuiZhouSmoke (被适配类),也实现了 Smoke 接口,在测试类中我们可以根据需要选择并创建任一种符合需求的子类,来实现具体功能。

当然,实际开发中可以参考SpringMvc的适配器。

Spring定义了一个适配接口,使得每一种Controller有一种对应的适配器实现类, 让适配器代替controller执行相应的方法。这样在扩展Controller 时,只需要增加一个适配器类就完成了SpringMVC的扩展了

五、对象适配器模式实现

另外一种适配器模式是对象适配器,它不是使用继承再实现的方式,而是使用直接关联,或者称为委托的方式。

1、目标接口

public interface Smoke {
    //烟草专卖
    public void sellSmoke();
}

2、适配者类

public class GuangdongSmoke implements Smoke{
    //广东省售烟平台在系统内,可以卖烟
    public void companyGD(){
        System.out.println("广东省烟草专卖");
    }

    @Override
    public void sellSmoke() {
        this.companyGD();
    }
}

public class GuiZhouSmoke {
    //贵州省没有接入系统,无法卖烟
    public void companyGZ(){
        System.out.println("贵州省接入烟草专卖");
    }

}

3、适配器类

这一次适配器类直接关联被适配类,同时实现标准接口

public class Adapter  implements Smoke {
    private GuiZhouSmoke guiZhouSmoke;

    public Adapter(GuiZhouSmoke guiZhouSmoke) {
        this.guiZhouSmoke = guiZhouSmoke;
    }

    /**
     *     适配器类,继承了被适配类,实现标准接口
     *     就相当于通过中间代码将贵州的烟草信息接入广州
      */
    @Override
    public void sellSmoke() {
       this.guiZhouSmoke.companyGZ();
    }
}

4、测试类

  public static void main(String[] args) {
        //本身功能实现
        Smoke gd =new GuangdongSmoke();
        gd.sellSmoke(); //广东省烟草专卖
        //通过适配器来实现
        GuiZhouSmoke guiZhouSmoke =new GuiZhouSmoke();
        Smoke gz =new Adapter(guiZhouSmoke);
        gz.sellSmoke();//贵州省接入烟草专卖
    }    

从IEDA的diagrams图也能看出来,只要Adapter自身拥有一个被适配类的对象,再把具体的特殊功能委托给这个对象来实现。使用对象适配器模式,可以使得 Adapter 类(适配类)根据传入的 Adaptee 对象达到适配多个不同被适配类的功能,当然,此时我们可以为多个被适配类提取出一个接口或抽象类。这样看起来的话,对象适配器模式更加灵活一点。

在 Spring 的 AOP 里通过使用的 Advice(通知)来增强被代理类的功能。Spring 实现这一 AOP 功能的原理就使用代理模式(1、JDK 动态代理。2、CGLib 字节码生成技术代理。)对类进行方法级别的切面增强,即,生成被代理类的代理类,并在代理类的方法前,设置拦截器,通过执行拦截器中的内容增强了代理方法的功能,实现的面向切面编程。

Advice(通知)的类型有:BeforeAdvice、AfterReturningAdvice、ThrowSadvice 等。每个类型 Advice(通知)都有对应的拦截器,MethodBeforeAdviceInterceptor、AfterReturningAdviceInterceptor、ThrowsAdviceInterceptor。Spring 需要将每个 Advice(通知)都封装成对应的拦截器类型,返回给容器,所以需要使用适配器模式对 Advice 进行转换。

六、适配器模式的应用

适配器模式主要应用于希望复用一些现存的类,但是接口又与复用环境要求不一致的情况。比如在需要对早期代码复用一些功能等应用上很有实际价值。适用场景大致包含三类:

  • 1、已经存在的类的接口不符合我们的需求
  • 2、创建一个可以复用的类,使得该类可以与其他不相关的类或不可预见的类(即那些接口可能不一定兼容的类)协同作
  • 3、在不对每一个都进行子类化以匹配它们的接口的情况下,使用一些已经存在的子类

适配器模式是对现存系统的封装,为现存系统提供一个更为方便的访问接口。适配器模式为事后设计,没有引入新的接口,只是将一个接口通过适配来间接转换为另一个接口。

李小庚听闻讲解后,从储物戒指拿出了不少法器(实时演算的),稍加构建,便形在丹田中成了一个个的接口,原本暗淡的法器突然变的流光溢彩起来。

“不过师弟记住,离开内网后,不要直接开始构建接口,接入法器的时候还是需要经过专家审核,防止出现一些冲突甚至引入一些致命破绽。”

“原来如此,谢谢师兄,我会给你五星好评的,不过我们云霄殿有专业的专家评审团,这一点你放心。”

“不用谢,狗大户。”