当前位置: 首页 > news >正文

4.结构型设计模式 - 第1回:引言与适配器模式 (Adapter Pattern) ——设计模式入门系列

一、引言

在现代软件开发中,设计模式是帮助我们解决复杂问题的工具,它们提供了在常见场景下重用已验证解决方案的途径。而结构型设计模式主要关注类与对象之间的组合方式,旨在通过增强灵活性和降低耦合度来改进代码的结构。

本次讨论的是结构型模式中的第一个:适配器模式 (Adapter Pattern)。它的核心目的是使接口不兼容的类能够协同工作,充当两个类之间的桥梁,保证代码的可扩展性和复用性。

二、适配器模式概述

1. 定义

适配器模式(Adapter Pattern)将一个类的接口转换为客户期望的另一个接口,使得原本接口不兼容的类可以一起工作。它常用于解决“现有接口”与“需要的接口”之间的不兼容问题。

2. 使用场景

适配器模式适用于以下场景:

  • 你希望使用一个现有的类,而它的接口并不符合你的需求。
  • 创建一个可以与多个不相关类协同工作的类,而不修改这些类的接口。

三、适配器模式的结构

适配器模式可以以类适配器和对象适配器两种方式实现:

  1. 类适配器:通过继承的方式实现适配,利用多继承模拟适配。
  2. 对象适配器:通过组合(持有被适配对象的实例)实现适配。
类图:
+--------------------------------+
|           客户端(Client)      |
+--------------------------------+| 使用目标接口v
+--------------------------------+
|           目标接口(Target)   |<-----+
+--------------------------------+      |^                           | 实现目标接口|                           |
+--------------------------------+       |
|           适配器(Adapter)    |-------+ 
| 适配器实现目标接口,并调用被适配对象 |
+--------------------------------+|v
+--------------------------------+
|        被适配者(Adaptee)     |
+--------------------------------+

四、代码实现

以下是一个简单的 Java 示例,展示如何将一个老旧的系统接口适配成新的系统接口。

1. 被适配类 (Adaptee)

// 现有类,无法直接改变
public class LegacyPrinter {public void printText(String text) {System.out.println("Legacy Printer: " + text);}
}

2. 目标接口 (Target)

// 客户端期望的接口
public interface Printer {void print(String message);
}

3. 适配器类 (Adapter)

// 通过适配器将旧的 LegacyPrinter 适配为 Printer 接口
public class PrinterAdapter implements Printer {private LegacyPrinter legacyPrinter;public PrinterAdapter(LegacyPrinter legacyPrinter) {this.legacyPrinter = legacyPrinter;}@Overridepublic void print(String message) {legacyPrinter.printText(message);}
}

4. 客户端代码 (Client)

public class Client {public static void main(String[] args) {Printer printer = new PrinterAdapter(new LegacyPrinter());printer.print("Hello, Adapter Pattern!");}
}

五、类适配器 vs 对象适配器

适配器类型特点优缺点
类适配器通过继承实现,适配器类同时继承了目标接口和被适配者类。优点:简洁,适合单一类适配。缺点:不支持多个类适配,受限于 Java 单继承机制。
对象适配器通过组合实现,适配器类包含一个被适配者类的实例,并实现目标接口。优点:更灵活,支持多个类的适配。缺点:稍微增加了间接层次。

六、案例分析

1. 现实生活案例:电源适配器

一个常见的现实生活中的例子是电源适配器。不同国家的电压标准和插头形状不同,电器设备无法直接使用。但通过电源适配器,电压和插头形状都能被转换为设备所需的标准,从而确保设备正常工作。

2. 综合案例:数据库连接适配器

假设我们有一个旧版数据库系统,它的连接接口已经过时,但现在我们希望在新的系统中使用不同的数据库连接接口。

// 旧版数据库系统
public class OldDatabase {public void connectToDb(String connectionString) {System.out.println("Connecting to database with " + connectionString);}
}// 新系统期望的数据库接口
public interface Database {void connect(String databaseUrl);
}// 适配器,将老接口适配为新接口
public class DatabaseAdapter implements Database {private OldDatabase oldDatabase;public DatabaseAdapter(OldDatabase oldDatabase) {this.oldDatabase = oldDatabase;}@Overridepublic void connect(String databaseUrl) {oldDatabase.connectToDb(databaseUrl);}
}// 客户端代码
public class Client {public static void main(String[] args) {Database db = new DatabaseAdapter(new OldDatabase());db.connect("jdbc:mysql://localhost:3306/mydb");}
}

七、补充与开发建议

在实际开发中,适配器模式是处理代码重构、引入第三方库或与遗留系统集成的有效手段。但在使用适配器时应注意以下几点:

  1. 谨慎过度使用:适配器模式容易滥用,如果系统中出现了大量的适配器类,可能预示着系统设计不够清晰。
  2. 适配的范围:当接口之间差异较大时,使用适配器可能会增加系统复杂性。最好将适配的范围控制在接口定义和调用的边界。
  3. 保持灵活性:为了保持代码的灵活性,可以结合其他模式,如工厂模式,通过工厂创建适配器实例。

八、结论

适配器模式是一种强大的模式,能够在保持原有类功能的同时,使其符合新的需求。通过合理使用,能够增强系统的兼容性和可扩展性。在架构演进和系统重构中,适配器模式是开发人员不可或缺的工具之一。


相关阅读

  • 设计模式入门系列

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 架构设计笔记-5-软件工程基础知识
  • 黑马智数Day2
  • 【protobuf】ProtoBuf的学习与使用⸺C++
  • Go语言语法基础
  • 蓝禾,汤臣倍健,三七互娱,得物,顺丰,快手,游卡,oppo,康冠科技,途游游戏,埃科光电25秋招内推
  • GNU链接器(LD):设置入口点(ENTRY命令)的用法及实例解析
  • 【python】requests 库 源码解读、参数解读
  • 使用 Python 模拟光的折射,反射,和全反射
  • 【技术解析】wx.request 封装:优化小程序网络请求的最佳实践
  • 人工智能面试题(Artificial Intelligence Algorithm Interview Questions)
  • 【深度学习】03-神经网络3-1梯度下降网络优化方法
  • 在Java中 String能存储多少个字符?
  • 前端——表单标签样式
  • 使用Python实现图形学曲线和曲面的B样条曲线算法
  • 通过 Xshell 无法连接到 Ubuntu
  • 【407天】跃迁之路——程序员高效学习方法论探索系列(实验阶段164-2018.03.19)...
  • 30秒的PHP代码片段(1)数组 - Array
  • Fastjson的基本使用方法大全
  • go append函数以及写入
  • python 装饰器(一)
  • Sequelize 中文文档 v4 - Getting started - 入门
  • vuex 学习笔记 01
  • 飞驰在Mesos的涡轮引擎上
  • 精彩代码 vue.js
  • 微信小程序实战练习(仿五洲到家微信版)
  • 线性表及其算法(java实现)
  • 一起来学SpringBoot | 第三篇:SpringBoot日志配置
  • 中文输入法与React文本输入框的问题与解决方案
  • 选择阿里云数据库HBase版十大理由
  • ​【原创】基于SSM的酒店预约管理系统(酒店管理系统毕业设计)
  • ​520就是要宠粉,你的心头书我买单
  • ​Spring Boot 分片上传文件
  • #define与typedef区别
  • #Linux(Source Insight安装及工程建立)
  • #vue3 实现前端下载excel文件模板功能
  • $.each()与$(selector).each()
  • (06)Hive——正则表达式
  • (16)Reactor的测试——响应式Spring的道法术器
  • (32位汇编 五)mov/add/sub/and/or/xor/not
  • (7)STL算法之交换赋值
  • (pytorch进阶之路)CLIP模型 实现图像多模态检索任务
  • (分布式缓存)Redis持久化
  • (蓝桥杯每日一题)平方末尾及补充(常用的字符串函数功能)
  • (四)c52学习之旅-流水LED灯
  • (四)汇编语言——简单程序
  • (转)JVM内存分配 -Xms128m -Xmx512m -XX:PermSize=128m -XX:MaxPermSize=512m
  • (转)使用VMware vSphere标准交换机设置网络连接
  • .dwp和.webpart的区别
  • .NET Core WebAPI中封装Swagger配置
  • .NET 通过系统影子账户实现权限维持
  • .Net 知识杂记
  • .NET成年了,然后呢?
  • .NET中的Event与Delegates,从Publisher到Subscriber的衔接!
  • @LoadBalanced 和 @RefreshScope 同时使用,负载均衡失效分析
  • @拔赤:Web前端开发十日谈