《Head First设计模式》第七章-适配器模式、外观模式

适配器模式

适配器模式是什么,你一定不难理解,因为现实中到处都是。比如说:
如果你需要在欧洲国家使用美国制造的笔记本电脑,你可能需要使用一个交流电的适配器……

图片说明
当你不想改变现有的代码,解决接口不适配问题,便可使用适配器模式,你可以写一个类,将新厂商接口转接成你所期望的接口。

图片说明

定义适配器模式:将一个类的接口,转换成客户期望的另一个接口。适配器让原本接口不兼容的类可以合作无间。

现在我们已经知道什么是适配器了,让我们后退一步,再次看看各部分之间的关系。

图片说明

类图:

图片说明

让我们以开头所提到的电脑电源插头适配的问题为例:

新建一个电脑类:

1

2

3

4

5

6

7

8

9

10

11

public class Computer {

     //充电方法只能使用两孔插座,只能传入两孔插座做参数

    public void charge(Socket_Two socket_Two) {

        socket_Two.connect();

        addPower();//调用增加电量的方法

    }

    //充电成功,电量增加

    private void addPower() {

        System.out.println("电源已连接,充电中...");

    }

创建一个两孔插座接口:

1

2

3

public interface Socket_Two {

    void connect();//连接插座方法

}

再创建一个三孔插座类:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

public class Socket_Three {   

    //连接方法

    public void connect() {

        //调用每个孔接通的方法

        leftConnect();

        rightconnect();

        extraConner();

    }

    //三孔接通方法

    public void rightconnect() {

        System.out.println("火线接通...");

    }

    public void leftConnect() {

        System.out.println("零线接通...");

    }

    public void extraConner() {

        System.out.println("地线接通...");

    }

}

创建一个适配器:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

//三孔插座与两孔插座适配器,直接关联被适配类,同时实现标准接口

public class LineWithSocket_Two implements Socket_Two {

    // 直接关联被适配类

    private Socket_Three socket_Three;

    //通过构造函数传入具体需要适配的被适配类对象

    public LineWithSocket_Two(Socket_Three socket_Three) {

        this.socket_Three=socket_Three;

    }

    @Override

    public void connect() {

        //使用委托的方式完成特殊功能

        System.out.println("我是适配器:通过我可以让两脚插头使用三孔插座")

        socket_Three.leftConnect();

        socket_Three.rightconnect();

    }

}

测试类

1

2

3

4

5

6

7

8

public static void main(String[] args) {

    Computer computer=new Computer();

    Socket_Three socket_three=new Socket_Three();

    //调用适配器类来完成适配

    LineWithSocket_Two lineWithScoket_Two=new LineWithSocket_Two(socket_three);

    System.out.println("使用适配器:");

    computer.charge(lineWithScoket_Two);

}

运行结果:

图片说明

其他

实际上适配器模式中有“两种”适配器:“对象”适配器“类”适配器

究竟什么是“类”适配器?为什么我们还没告诉你这种适配器?因为你需要多重继承才能够实现它,这在Java中是不可能的。但是当你在使用多重继承语言的时候,还是可能遇到这样的需求。

让我们看看多重继承的类图:

图片说明

看起来很熟悉吗?没错,唯一的差别就在于适配器继承了Target和Adaptee。而对象适配器利用组合的方式将请求传送给被适配者。

对象适配器和类适配器使用两种不同的适配方法(分别是组合与继承)。

适配器模式适用场景

  1. 重复使用现有的类,而此类的接口不符合系统的需要。在遗留代码复用、类库迁移等方面非常有用。
  2. 想要建立一个可以重用的类,用于与一些彼此之间没有太大关联的一些类,包括一些可能在将来引进的类一起工作。
  3. 使用第三方组件或中间件,组件接口定义和自己定义的不同,不希望修改自己的接口,但是要使用第三方组件接口的 功能,避免重复造轮子。

适配器模式优缺点

优点:

  1. 将目标类和适配者类解耦。
  2. 增加了类的透明性和复用性,将具体的实现封装在适配者类中,对于客户端类来说是透明的,提高了适配者的复用性。
  3. 灵活性和扩展性都非常好,符合开闭原则

类适配器优点:

  • 由于适配器类是适配者类的子类,因此可以在适配器类中置换一些适配者的方法,使得适配器的灵活性更强。

对象适配器优点:

  • 把多个不同的适配者适配到同一个目标,也就是说,同一个适配器可以把适配者类和他的子类都适配到目标接口。

缺点:

  • 实现适配器所需要的工作和目标接口的大小成正比,接口越复杂适配器也越复杂。

类适配器缺点:

  • 对于Java、C#等不支持多重继承的语言,一次最多只能适配一个适配者类,而且目标抽象类只能为接口,不能为类,其使用有一定的局限性,不能将一个适配者类和他的子类同时适配到目标接口。

对象适配器的缺点:

  • 与类适配器模式相比,要想置换适配者类的方法就不容易。

 




通过上一篇你已经知道适配器模式是如何将一个类的接口转换成另一个符合客户期望的接口的。你也知道在Java中要做到这一点,必须将一个不兼容接口的对象包装起来,变成兼容的对象。
​ 我们现在要看一个改变接口的新模式,但是它改变接口的原因是为了简化接口,这个模式被巧妙的命名为外观模式。它将一个或数个类的复杂的一切都隐藏在背后,只显露出一个干净美好的外观。

定义

外观模式定义:外观模式提供了一个统一的接口,用来访问子系统中的一群接口。外观定义了一个高层接口,让子系统更容易使用。

外观模式实现了最少知识原则(Least Knowledge principle),这个原则希望不要让太多的类耦合在一起,对用户来说只和一个外观类打交道了,达到客户和一群子系统的解耦。

例题:搭建一个家庭影院系统
系统内包含设备:DVD播放器、投影机、自动屏幕、立体声音响、爆米花机........
来看看这些组件的组成:
图片说明

当你把所有设备布置好后,准备看电源时...你忘了你必须要先一个个启用这些设备...关闭时也还将要进行一遍反向操作(崩溃....)。
图片说明
结果你发现要使用你的家庭影院是那么的麻烦!

因此,我们引入外观模式,有了外观模式,通过实现一个更合理的接口的外观类,你可以将一个复杂的子系统变的容易使用。看看改变之后的类图:

图片说明

定义这些媒体类:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

class Amplifier{

    public void on(){

        System.out.println("欢迎使用功放类。。");

    }

    public void off(){

        System.out.println("已经关闭功放。。");

    }

    public void setCD(){

        System.out.println("正在安放CD。。。");

    }

    public void setDVD(){

        System.out.println("正在安放DVD、。。");

    }

    public void setStereoSound(){

        System.out.println("设置立体声。。");

    }

    public void setSurroundSound(){

        System.out.println("设置环绕立体声。。");

    }

    public void setTime(){

        System.out.println("正在设置时间。。");

    }

    public void setVolume(){

        System.out.println("正在设置音量。。");

    }

}

//定义Tuner类

class Tuner{

    public void on(){

        System.out.println("正在 打开调谐器。。");

    }

    public void off(){

        System.out.println("正在关闭调谐器。。");

    }

    public void setAM(){

        System.out.println("正在设置am。。");

    }

    public void setFM(){

        System.out.println("正在设置频道。。");

    }

    public void setFrequency(){

        System.out.println("正咋设置频道。。");

    }

}

//定义DVD播放器类

class DVDPlayer{

    public void on(){

        System.out.println("正在打开DVD。。");

    }

    public void off(){

        System.out.println("正在关闭DVD。。");

    }

    public void pause(){

        System.out.println("已经暂停DVD播放。。");

    }

    public void play(){

        System.out.println("正在播放DVD。。");

    }

    public void setTwoChannelAudio(){

        System.out.println("正在设置双频道。。");

    }

    public void setSurroundAudio(){

        System.out.println("正在设置环绕立体声。。");

    }

}

//定义CD播放器

class CDPlayer{

    public void on(){

        System.out.println("正在打开CD");

    }

    public void off(){

        System.out.println("正在关闭CD");

    }

    public void eject(){

        System.out.println("弹出CD播放器!");

    }

    public void pause(){    

    }

    public void play(){

    }

    public String toString(){

        return "hello panda";

    }

}

//定义投影仪

class Projector{

    public void on(){

        System.out.println("正在打开投影仪。。");

    }

    public void off(){

        System.out.println("正在关闭投影仪。。");

    }

    public void setTVMode(){

        System.out.println("正在设置tv模式。。");

    }

    public void setWideScreenMode(){

        System.out.println("正在设置宽屏模式。。");

    }

}

//定义屏幕

class Screen{

    public void up(){

        System.out.println("正在生起屏幕。。");

    }

    public void down(){

        System.out.println("正在放下屏幕。。");

    }

}

//定义爆米花机

class PopcornPopper{

    public void on(){

        System.out.println("正在打开爆米花机。。");

    }

    public void off(){

        System.out.println("正在关闭爆米花机。。");

    }

    public  void pop(){

        System.out.println("正在蹦爆米花。。");

    }

}

//定义影院灯光

class TheaterLights{

    public void on(){

        System.out.println("正在打开灯光。。");

    }

    public void off(){

        System.out.println("正在关闭灯光。。");

    }

    public void dim(){

        System.out.println("正在调暗灯光。。");

    }

 

}

多媒体的具体类准备好就可以定义外观模式类了。

外观模式类:俩方法,一个看电影-打开一系列设备,一个电影结束-关闭一系列设备

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

//定义外观模式家庭影院

class HomeTheaterFacade{

    Amplifier amp;

    Tuner tuner ;

    DVDPlayer dvd;

    CDPlayer cd;

    Projector project;

    TheaterLights light;

    Screen screen;

    PopcornPopper pop;

    //构造的时候拿到这些对象

    public HomeTheaterFacade(Amplifier amp,Tuner tuner,DVDPlayer dvd, 

            CDPlayer cd ,Projector project,TheaterLights light,Screen screen

            ,PopcornPopper pop){

        this.amp = amp;

        this.tuner = tuner;

        this.dvd = dvd;

        this.cd = cd;

        this.project = project;

        this.light  = light;

        this.screen = screen;

        this.pop = pop;

    }

    //看电影  放一个方法里来执行一系列动作

    public void watchMovie(String movie){

        System.out.println("get ready to watch a movie..");

        pop.on();//首先打开爆米花机

        pop.pop();//然后蹦爆米花

        light.dim();//把灯光调暗

        screen.down();//投影仪放下来

        project.on();

        project.setWideScreenMode();

        amp.on();

        amp.setDVD();

        amp.setSurroundSound();

        amp.setVolume();

        dvd.on();

        dvd.play();

    }

    //电影结束

    public void endMovie(String movie){

        System.out.println("shutting movie theater down..");

        pop.off();

        light.on();

        screen.up();

        project.off();

        amp.off();

       dvd.off();

    }

}

测试

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

public class FacadePattern {

    public static void main(String args[]){

        Amplifier amp = new Amplifier();

        Tuner tuner = new Tuner() ;

        DVDPlayer dvd = new DVDPlayer();

        CDPlayer cd = new CDPlayer();

        Projector project = new Projector();;

        TheaterLights light = new TheaterLights();

        Screen screen = new Screen();

        PopcornPopper pop = new PopcornPopper();                  

        HomeTheaterFacade facade = new HomeTheaterFacade(amp,tuner,dvd,cd,project,light,screen,pop);

        facade.watchMovie("movie");

        facade.endMovie("movie");

    }

 

}

通过上面的代码可以看出,每个类对象都要执行一些方法,如果直接new这些类创建对象去调方法会与这些类产生耦合,这时单独再写一个外观类,构造初始化时拿到这些类对象,在一个方法里去调这些类对象的方法,这样对客户来说只和一个类打交道,与子系统的一堆类解耦了。此类原则即为:最少知识原则

最少知识原则:只和你的密友谈话。

客户端只和外观谈话,不和子设备,如DVD播放机、投影仪等谈话,降低了客户端和设备的耦合度。

下面给出最少知识原则的指导思想:

就任何对象而言,在该对象的方法内,我们只应该调用属于以下范围的方法:

1.该对象本身

2.被当做方法的参数而传过来的对象

3.该方法所创建或实例化的任何对象

4.对象的任何组件

尽可能自己封装子设备的方法,以便减少客户端和子设备的耦合。

外观模式优点

松耦合:使客户端与子系统之间解耦,让子系统内部的模块功能更容易扩展与维护。

简单易用:客户端无需了解子系统的内部实现及内部构成,只需要与外观类交互即可。

更好地划分访问层次:部分方法对外,部分方法对内交互使用。子系统将暴露在外的功能集中到外观类中可以隐藏子系统内部细节。

适配器、装饰者、外观模式对比

  • 在介绍装饰者模式时,引出了一个开闭原则,即对修改关闭,对扩展开放。装饰者模式主要强调的是在不改变原有类的基础上,添加新功能。
  • 适配器模式主要是对适配对象进行调整,以便适合消费者的需求。从而达到消费者和被适配者解耦的目的。
  • 外观模式的特点主要是简化接口,以及减少客户端对外观组件的耦合。因为如果客户端变化来,组件的子系统变化了,不用影响客户端。除此之外,在封装组件时,适当的在外观类中添加一些自己想要的规则。如上面例子中各设备的开关顺序

总结

  • 当需要使用一个现有的类,而其接口并不符合你的需要时,就使用适配器。
  • 当需要简化并统一一个很大的接口或者一群复杂的接口时,使用外观。
  • 适配器改变接口以符合客户的希望
  • 外观将一个客户从复杂的子系统中解耦。
  • 实现一个适配器可能需要一番功夫,也可能不费工夫,视目标接口的大小与复杂度而定。
  • 实现一个外观,需要将子系统组合进外观中,然后将工作委托给子系统执行。
  • 适配器模式有两种形式,对象适配器和类适配器。类适配器需要用到多重继承。
  • 你可以为一个子系统实现一个以上的外观。
  • 适配器将一个对象包装起来以改变其接口;装饰器将一个对象包装起来以增加新的行为和责任,而外观将一群对象包装起来以简化其接口。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/445420.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《Head First设计模式》第八章笔记-模板方法模式

模板方法模式 之前所学习的模式都是围绕着封装进行,如对象创建、方法调用、复杂接口的封装等,这次的模板方法模式将深入封装算法块,好让子类可以在任何时候都将自己挂接进运算里。 模板方法定义:模板方法模式在一个方法中定义一…

机器学习基础-吴恩达-coursera-(第一周学习笔记)----Introduction and Linear Regression

课程网址:https://www.coursera.org/learn/machine-learning Week 1 —— Introduction and Linear Regression 目录 Week 1 Introduction and Linear Regression目录一 介绍1-1 机器学习概念及应用1-2 机器学习分类 二 单变量的线性回归2-1 假设函数hypothesis2…

常见8种机器学习算法总结

简介 机器学习算法太多了,分类、回归、聚类、推荐、图像识别领域等等,要想找到一个合适算法真的不容易,所以在实际应用中,我们一般都是采用启发式学习方式来实验。通常最开始我们都会选择大家普遍认同的算法,诸如SVM&a…

redis——数据结构(字典、链表、字符串)

1 字符串 redis并未使用传统的c语言字符串表示,它自己构建了一种简单的动态字符串抽象类型。 在redis里,c语言字符串只会作为字符串字面量出现,用在无需修改的地方。 当需要一个可以被修改的字符串时,redis就会使用自己实现的S…

Hotspot虚拟机的对象

创建 Step1:类加载检查 虚拟机遇到一条 new 指令时,首先将去检查这个指令的参数是否能在常量池中定位到这个类的符号引用,并且检查这个符号引用代表的类是否已被加载过、解析和初始化过。如果没有,那必须先执行相应的类加载过程。 Step2:分…

redis——数据结构(整数集合,压缩列表)

4、整数集合 整数集合(intset)是 Redis 用于保存整数值的集合抽象数据结构, 可以保存 int16_t 、 int32_t 、 int64_t 的整数值, 并且保证集合中不会出现重复元素。 实现较为简单: typedef struct intset {// 编码方…

机器学习知识总结系列- 知识图谱(0-0)

文章目录目录机器学习知识图谱目录 本系列的文章只是根据个人的习惯进行总结,可能结构与一些书籍上不太一样,开始的内容比较简单,会随着后续的深入,不断丰富和更新图谱,同时也期待有相同兴趣的朋友一起给我留言一起丰富…

跳表介绍和实现

想慢慢的给大家自然的引入跳表。 想想,我们 1)在有序数列里搜索一个数 2)或者把一个数插入到正确的位置 都怎么做? 很简单吧 对于第一个操作,我们可以一个一个比较,在数组中我们可以二分,这…

机器学习知识总结系列- 基本概念(1-0)

文章目录目录1. 机器学习的定义2. 机器学习的分类2.1根据是否在人类监督下进行训练监督学习非监督学习半监督学习强化学习2.2根据是否可以动态渐进的学习在线学习批量学习2.3根据是否在训练数据过程中进行模式识别实例学习基于模型的学习3. 机器学习中的一些常见名词4. 机器学习…

剑指offer(刷题21-30)--c++,Python版本

文章目录目录第 21题:解题思路:代码实现:cpython第22 题:解题思路:代码实现:cpython第23 题:解题思路:代码实现:cpython第24 题:解题思路:代码实现…

剑指offer(刷题41-50)--c++,Python版本

文章目录目录第41题:解题思路:代码实现:cpython第42题:解题思路:代码实现:cpython第43题:解题思路:代码实现:cpython第44题:解题思路:代码实现&am…

redis——持久化

因为redis是内存数据库,他把数据都存在内存里,所以要想办法实现持久化功能。 RDB RDB持久化可以手动执行,也可以配置定期执行,可以把某个时间的数据状态保存到RDB文件中,反之,我们可以用RDB文件还原数据库…

剑指offer(刷题51-60)--c++,Python版本

文章目录目录第51题:解题思路:代码实现:cpython第52题:解题思路:代码实现:cpython第53题:解题思路:代码实现:cpython第54题:解题思路:代码实现&am…

2017第一届河北省大学生程序设计竞赛题解

超级密码 小明今年9岁了,最近迷上了设计密码!今天,他又设计了一套他认为很复杂的密码,并且称之为“超级密码”. 说实话,这套所谓的“超级密码”其实并不难:对于一个给定的字符串,你只要提取其中…

大数的四则运算(加法、减法、乘法、除法)

大数的四则运算(加法、减法、乘法、除法) 前言: 在计算机中数字表示的范围是有限制的,比如我们熟知的 int、float、double 等数据类型所能表示的范围都是有限的,如果我们要对位数达到几十位、几百位、上千位的大整数进…

随机过程1

随机过程1概述1.参考书目2.主要内容3.概率论--基本概念回顾3.1对“不确定性”的认识3.2 应对“不确定性”应该怎么做3.3随机变量(Random Variable)3.4分布函数(Distribution Function)3.5概率密度(Density)…

数组基操三连(4)

题目一 给定一个长度为N的整型数组arr,其中有N个互不相等的自然数1~N 请实现arr的排序 但是不要把下标0~N-1位置上的数值通过直接赋值的方式替换成1~N。 要求:时间复杂度为O(N),额外空间复杂度为O(1)。 思路:从左向右检查&…

Linux(1)-touch,mkdir,rm,mv,cp,ls,cd,cat

Linux1-实用终端命令1. touch, mkdir2. rm, mv, cp3. ls(通配符),cd(绝对/相对路径)4. cat, more/less文件内容浏览文件/目录-增删查改, 文件内容查看.1. touch, mkdir touch新文件 :在当前文件夹下,创建文件。文件不存在则创建新文件;文件存…