结构型模式

本贴最后更新于 709 天前,其中的信息可能已经事过景迁

结构型模式主要涉及如何组合各种对象以便获得更好、更灵活的结构。虽然面向对象的继承机制提供了最基本的子类扩展父类的功能,但结构型模式不仅仅简单地使用继承,而更多地通过组合与运行期的动态组合来实现更灵活的功能。

适配器


将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。

适配器模式是 Adapter,也称 Wrapper,是指如果一个接口需要 B 接口,但是待传入的对象却是 A 接口,怎么办?

我们举个例子。如果去美国,我们随身带的电器是无法直接使用的,因为美国的插座标准和中国不同,所以,我们需要一个适配器:

adapter

在程序设计中,适配器也是类似的。我们已经有一个 Task​类进行一个 1 到 num 的求和,实现了 Callable​接口:

public class Task implements Callable<Long> { private long num; public Task(long num) { this.num = num; } public Long call() throws Exception { long r = 0; for (long n = 1; n <= this.num; n++) { r = r + n; } System.out.println("Result: " + r); return r; } }

现在,我们想通过一个线程去执行它:

Callable<Long> callable = new Task(123450000L); Thread thread = new Thread(callable); // compile error! thread.start();

发现编译不过!因为 Thread 接收 Runnable 接口,但不接收 Callable 接口,肿么办?

一个办法是改写 Task 类,把实现的 Callable 改为 Runnable,但这样做不好,因为 Task 很可能在其他地方作为 Callable 被引用,改写 Task 的接口,会导致其他正常工作的代码无法编译。

另一个办法不用改写 Task 类,而是用一个 Adapter,把这个 Callable 接口“变成”Runnable 接口,这样,就可以正常编译:

Callable<Long> callable = new Task(123450000L); Thread thread = new Thread(new RunnableAdapter(callable)); thread.start();

这个 RunnableAdapter 类就是 Adapter,它接收一个 Callable,输出一个 Runnable。怎么实现这个 RunnableAdapter 呢?我们先看完整的代码:

public class RunnableAdapter implements Runnable { // 引用待转换接口: private Callable<?> callable; public RunnableAdapter(Callable<?> callable) { this.callable = callable; } // 实现指定接口: public void run() { // 将指定接口调用委托给转换接口调用: try { callable.call(); } catch (Exception e) { throw new RuntimeException(e); } } }

编写一个 Adapter 的步骤如下:

  1. 实现目标接口,这里是 Runnable
  2. 内部持有一个待转换接口的引用,这里是通过字段持有 Callable 接口;
  3. 在目标接口的实现方法内部,调用待转换接口的方法。

这样一来,Thread 就可以接收这个 RunnableAdapter,因为它实现了 Runnable 接口。Thread 作为调用方,它会调用 RunnableAdapterrun() 方法,在这个 run() 方法内部,又调用了 Callablecall() 方法,相当于 Thread 通过一层转换,间接调用了 Callablecall() 方法。

适配器模式在 Java 标准库中有广泛应用。比如我们持有数据类型是 String[],但是需要 List 接口时,可以用一个 Adapter:

String[] exist = new String[] {"Good", "morning", "Bob", "and", "Alice"}; Set<String> set = new HashSet<>(Arrays.asList(exist));

注意到 List<T> Arrays.asList(T[]) 就相当于一个转换器,它可以把数组转换为 List

我们再看一个例子:假设我们持有一个 InputStream,希望调用 readText(Reader) 方法,但它的参数类型是 Reader 而不是 InputStream,怎么办?

当然是使用适配器,把 InputStream“变成”Reader

InputStream input = Files.newInputStream(Paths.get("/path/to/file")); Reader reader = new InputStreamReader(input, "UTF-8"); readText(reader);

InputStreamReader 就是 Java 标准库提供的 Adapter,它负责把一个 InputStream 适配为 Reader。类似的还有 OutputStreamWriter

如果我们把 readText(Reader) 方法参数从 Reader 改为 FileReader,会有什么问题?这个时候,因为我们需要一个 FileReader 类型,就必须把 InputStream 适配为 FileReader

FileReader reader = new InputStreamReader(input, "UTF-8"); // compile error!

直接使用 InputStreamReader 这个 Adapter 是不行的,因为它只能转换出 Reader 接口。事实上,要把 InputStream 转换为 FileReader 也不是不可能,但需要花费十倍以上的功夫。这时,面向抽象编程这一原则就体现出了威力:持有高层接口不但代码更灵活,而且把各种接口组合起来也更容易。一旦持有某个具体的子类类型,要想做一些改动就非常困难。

小结

Adapter 模式可以将一个 A 接口转换为 B 接口,使得新的对象符合 B 接口规范。

编写 Adapter 实际上就是编写一个实现了 B 接口,并且内部持有 A 接口的类:

public BAdapter implements B { private A a; public BAdapter(A a) { this.a = a; } public void b() { a.a(); } }

在 Adapter 内部将 B 接口的调用“转换”为对 A 接口的调用。

只有 A、B 接口均为抽象接口时,才能非常简单地实现 Adapter 模式。

  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    201 引用 • 120 回帖 • 3 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • 分享

    有什么新发现就分享给大家吧!

    248 引用 • 1794 回帖
  • B3log

    B3log 是一个开源组织,名字来源于“Bulletin Board Blog”缩写,目标是将独立博客与论坛结合,形成一种新的网络社区体验,详细请看 B3log 构思。目前 B3log 已经开源了多款产品:SymSoloVditor思源笔记

    1062 引用 • 3455 回帖 • 151 关注
  • 酷鸟浏览器

    安全 · 稳定 · 快速
    为跨境从业人员提供专业的跨境浏览器

    3 引用 • 59 回帖 • 54 关注
  • PWA

    PWA(Progressive Web App)是 Google 在 2015 年提出、2016 年 6 月开始推广的项目。它结合了一系列现代 Web 技术,在网页应用中实现和原生应用相近的用户体验。

    14 引用 • 69 回帖 • 184 关注
  • jQuery

    jQuery 是一套跨浏览器的 JavaScript 库,强化 HTML 与 JavaScript 之间的操作。由 John Resig 在 2006 年 1 月的 BarCamp NYC 上释出第一个版本。全球约有 28% 的网站使用 jQuery,是非常受欢迎的 JavaScript 库。

    63 引用 • 134 回帖 • 733 关注
  • CodeMirror
    2 引用 • 17 回帖 • 174 关注
  • JetBrains

    JetBrains 是一家捷克的软件开发公司,该公司位于捷克的布拉格,并在俄国的圣彼得堡及美国麻州波士顿都设有办公室,该公司最为人所熟知的产品是 Java 编程语言开发撰写时所用的集成开发环境:IntelliJ IDEA

    18 引用 • 54 回帖
  • Thymeleaf

    Thymeleaf 是一款用于渲染 XML/XHTML/HTML5 内容的模板引擎。类似 Velocity、 FreeMarker 等,它也可以轻易的与 Spring 等 Web 框架进行集成作为 Web 应用的模板引擎。与其它模板引擎相比,Thymeleaf 最大的特点是能够直接在浏览器中打开并正确显示模板页面,而不需要启动整个 Web 应用。

    11 引用 • 19 回帖 • 395 关注
  • 链滴

    链滴是一个记录生活的地方。

    记录生活,连接点滴

    183 引用 • 3885 回帖
  • danl
    179 关注
  • FFmpeg

    FFmpeg 是一套可以用来记录、转换数字音频、视频,并能将其转化为流的开源计算机程序。

    23 引用 • 32 回帖 • 8 关注
  • V2Ray
    1 引用 • 15 回帖 • 2 关注
  • Solidity

    Solidity 是一种智能合约高级语言,运行在 [以太坊] 虚拟机(EVM)之上。它的语法接近于 JavaScript,是一种面向对象的语言。

    3 引用 • 18 回帖 • 445 关注
  • PHP

    PHP(Hypertext Preprocessor)是一种开源脚本语言。语法吸收了 C 语言、 Java 和 Perl 的特点,主要适用于 Web 开发领域,据说是世界上最好的编程语言。

    167 引用 • 408 回帖 • 482 关注
  • WebComponents

    Web Components 是 W3C 定义的标准,它给了前端开发者扩展浏览器标签的能力,可以方便地定制可复用组件,更好的进行模块化开发,解放了前端开发者的生产力。

    1 引用 • 14 关注
  • 星云链

    星云链是一个开源公链,业内简单的将其称为区块链上的谷歌。其实它不仅仅是区块链搜索引擎,一个公链的所有功能,它基本都有,比如你可以用它来开发部署你的去中心化的 APP,你可以在上面编写智能合约,发送交易等等。3 分钟快速接入星云链 (NAS) 测试网

    3 引用 • 16 回帖 • 1 关注
  • 人工智能

    人工智能(Artificial Intelligence)是研究、开发用于模拟、延伸和扩展人的智能的理论、方法、技术及应用系统的一门技术科学。

    115 引用 • 319 回帖
  • Node.js

    Node.js 是一个基于 Chrome JavaScript 运行时建立的平台, 用于方便地搭建响应速度快、易于扩展的网络应用。Node.js 使用事件驱动, 非阻塞 I/O 模型而得以轻量和高效。

    139 引用 • 269 回帖 • 2 关注
  • Firefox

    Mozilla Firefox 中文俗称“火狐”(正式缩写为 Fx 或 fx,非正式缩写为 FF),是一个开源的网页浏览器,使用 Gecko 排版引擎,支持多种操作系统,如 Windows、OSX 及 Linux 等。

    7 引用 • 30 回帖 • 372 关注
  • Anytype
    3 引用 • 31 回帖 • 28 关注
  • 创业

    你比 99% 的人都优秀么?

    81 引用 • 1395 回帖
  • BookxNote

    BookxNote 是一款全新的电子书学习工具,助力您的学习与思考,让您的大脑更高效的记忆。

    笔记整理交给我,一心只读圣贤书。

    1 引用 • 1 回帖 • 3 关注
  • V2EX

    V2EX 是创意工作者们的社区。这里目前汇聚了超过 400,000 名主要来自互联网行业、游戏行业和媒体行业的创意工作者。V2EX 希望能够成为创意工作者们的生活和事业的一部分。

    16 引用 • 236 回帖 • 240 关注
  • 反馈

    Communication channel for makers and users.

    120 引用 • 906 回帖 • 281 关注
  • CAP

    CAP 指的是在一个分布式系统中, Consistency(一致性)、 Availability(可用性)、Partition tolerance(分区容错性),三者不可兼得。

    12 引用 • 5 回帖 • 633 关注
  • Tomcat

    Tomcat 最早是由 Sun Microsystems 开发的一个 Servlet 容器,在 1999 年被捐献给 ASF(Apache Software Foundation),隶属于 Jakarta 项目,现在已经独立为一个顶级项目。Tomcat 主要实现了 JavaEE 中的 Servlet、JSP 规范,同时也提供 HTTP 服务,是市场上非常流行的 Java Web 容器。

    162 引用 • 529 回帖 • 9 关注
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    127 引用 • 169 回帖