【再谈设计模式】适配器模式 ~接口兼容的桥梁

一、引言

        在软件开发的复杂世界里,不同的组件、类或者系统往往有着各自独立的设计和接口定义。当需要将这些原本不兼容的部分整合在一起协同工作时,就像尝试将方形的榫头插入圆形的卯眼一样困难。适配器设计模式就如同一位神奇的工匠,能够巧妙地解决这个问题,让不同接口之间实现无缝对接。

二、定义与描述

        适配器设计模式属于结构型设计模式。它的主要作用是将一个类的接口转换为另一个接口,使原本由于接口不兼容而不能一起工作的类能够协同工作。可以把适配器想象成一个中间件,它包裹着一个已有的类,对外提供一个符合目标需求的新接口。

三、抽象背景

        在大型软件项目中,往往会集成多个不同的库或者模块。这些模块可能是由不同的团队开发,或者是在不同的时期基于不同的需求开发的。每个模块都有自己的接口设计,当需要将它们组合使用时,就会出现接口不匹配的情况。例如,一个旧的数据库访问模块可能提供了一种特定的查询接口,而新的业务逻辑层需要一种不同格式的查询结果。这时候就需要适配器来协调两者之间的差异。

四、适用场景与现实问题解决

1、适用场景

集成第三方库
        当使用第三方库时,其接口可能与项目中的其他部分不兼容。例如,一个图形绘制库的坐标系统与项目中自定义的坐标系统不同,通过适配器可以将两者协调起来。
旧系统升级
        在对旧系统进行升级时,新的模块可能采用了新的接口标准。使用适配器模式可以让旧系统中的部分继续使用,而不必对旧系统进行大规模的重写。
多平台适配


        对于需要在不同平台(如Windows、Linux、Mac)上运行的软件,不同平台可能有不同的API。适配器可以将不同平台的API转换为统一的接口,使业务逻辑层能够在不同平台上无缝运行。

2、现实问题解决

        假设一个公司收购了另一家公司,被收购公司有一套已经开发好的用户认证系统。收购公司的主系统有自己的用户认证接口,通过适配器模式,可以将被收购公司的用户认证系统适配到收购公司的主系统中,避免重新开发用户认证功能,节省时间和成本。

五、现实生活的例子

1、电源适配器

        不同国家的电源插座标准不同,如中国的插座是扁头的,而一些国外的电器插头可能是圆头的。电源适配器就起到了接口转换的作用,它可以将国内的电源接口转换为适合国外电器使用的接口,反之亦然。

2、手机充电器转接头

        新的手机可能采用了新的充电接口标准,但是旧的充电器仍然可以使用,只需要一个转接头(适配器),将旧充电器的接口转换为新手机能够接受的接口。

六、初衷与问题解决

        初衷是为了提高软件的可复用性和灵活性。在面对接口不兼容的情况时,不必修改原有的类或模块,通过适配器可以快速地解决接口匹配问题,降低了代码的耦合度,使得系统更容易维护和扩展。

七、代码示例

Java示例

// 目标接口
interface Target {
    void request();
}

// 被适配的类
class Adaptee {
    public void specificRequest() {
        System.out.println("Adaptee's specific request");
    }
}

// 适配器类
class Adapter implements Target {
    private Adaptee adaptee;

    public Adapter(Adaptee adaptee) {
        this.adaptee = adaptee;
    }

    @Override
    public void request() {
        adaptee.specificRequest();
    }
}

public class Main {
    public static void main(String[] args) {
        Adaptee adaptee = new Adaptee();
        Target target = new Adapter(adaptee);
        target.request();
    }
}

C++示例

#include <iostream>

// 目标类(抽象类)
class Target {
public:
    virtual void request() = 0;
    virtual ~Target() {}
};

// 被适配的类
class Adaptee {
public:
    void specificRequest() {
        std::cout << "Adaptee's specific request" << std::endl;
    }
};

// 适配器类
class Adapter : public Target {
private:
    Adaptee* adaptee;
public:
    Adapter(Adaptee* adaptee) : adaptee(adaptee) {}
    void request() override {
        adaptee->specificRequest();
    }
    ~Adapter() {
        delete adaptee;
    }
};

int main() {
    Adaptee* adaptee = new Adaptee();
    Target* target = new Adapter(adaptee);
    target.request();
    delete target;
    return 0;
}

Python示例

# 目标接口
class Target:
    def request(self):
        pass


# 被适配的类
class Adaptee:
    def specific_request(self):
        print("Adaptee's specific request")


# 适配器类
class Adapter(Target):
    def __init__(self, adaptee):
        self.adaptee = adaptee

    def request(self):
        self.adaptee.specific_request()


if __name__ == "__main__":
    adaptee = Adaptee()
    target = Adapter(adaptee)
    target.request()

Go示例

package main

import "fmt"

// 目标接口
type Target interface {
    request()
}

// 被适配的类
type Adaptee struct{}

func (a *Adaptee) specificRequest() {
    fmt.Println("Adaptee's specific request")
}

// 适配器结构体
type Adapter struct {
    adaptee *Adaptee
}

func (a *Adapter) request() {
    a.adaptee.specificRequest()
}

func main() {
    adaptee := &Adaptee{}
    target := &Adapter{adaptee: adaptee}
    target.request()
}

八、适配器设计模式的优缺点

优点

提高了代码的复用性
        可以复用现有的类,而不需要修改它们的代码。通过适配器将其适配到新的接口下,就可以在新的场景中使用。
降低了代码的耦合度
        被适配的类和使用适配后接口的类之间不需要直接交互,它们通过适配器进行通信。这样,当其中一方发生变化时,只要适配器的逻辑不变,另一方就不需要修改。
灵活性好
        可以很容易地替换适配器或者被适配的类,只要遵循相应的接口规范。

缺点

增加了代码的复杂性
        如果过多地使用适配器模式,会使得代码结构变得复杂,增加了理解和维护的难度。因为需要理解适配器、被适配的类以及目标接口之间的关系。
可能会降低性能
        由于适配器在中间做了一层转换,可能会对性能产生一定的影响,尤其是在对性能要求极高的场景下。

九、适配器设计模式的升级版

双向适配器

        普通的适配器是单向的,即从被适配的类转换到目标接口。双向适配器则可以实现双向的转换,既能将被适配类转换为目标接口,也能将符合目标接口的对象转换为被适配类的接口。例如,在两个不同的库之间,不仅要让A库能在B库的接口下工作,也要让B库能在A库的接口下工作,就可以使用双向适配器。

对象适配器和类适配器的混合使用

        在某些复杂的场景下,可以结合对象适配器(使用对象组合来实现适配)和类适配器(使用继承来实现适配)的优点。例如,先通过类适配器继承一些基本的功能,再通过对象组合来实现更灵活的适配。这种混合方式可以根据具体的需求,在代码复用性、灵活性和性能之间取得更好的平衡。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

光猫、路由器、交换机之连接使用(Connection and Usage of Optical Cats, Routers, and Switches)

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 本人主要分享计算机核心技…

《深入理解 Spring MVC 工作流程》

一、Spring MVC 架构概述 Spring MVC 是一个基于 Java 的轻量级 Web 应用框架&#xff0c;它遵循了经典的 MVC&#xff08;Model-View-Controller&#xff09;设计模式&#xff0c;将请求、响应和业务逻辑分离&#xff0c;从而构建出灵活可维护的 Web 应用程序。 在 Spring MV…

【Python小技巧】高效实现文件批量重命名

&#x1f9d1; 博主简介&#xff1a;曾任某智慧城市类企业算法总监&#xff0c;目前在美国市场的物流公司从事高级算法工程师一职&#xff0c;深耕人工智能领域&#xff0c;精通python数据挖掘、可视化、机器学习等&#xff0c;发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…

ES分词环境实战

文章目录 安装下载1.1 下载镜像1.2 单节点启动 防火墙设置异常处理【1】iptable链路中断 参考文档 参加完2024年11月软考&#xff0c;对ES的分词进行考查&#xff0c;前期有【 Docker 环境下安装部署 Elasticsearch 和 kibana】和【 Docker 环境下为 Elasticsearch 安装IK 分…

华为云stack网络服务流量走向

1.同VPC同子网同主机内ECS间互访流量走向 一句话通过主机内部br-int通信 2.同VPC同子网跨主机ECS间互访流量走向 3.同VPC不同子网同主机ECS间互访流量走向 去往本机的mac地址都记录在br-tun流表里 4.同VPC不同子网跨主机ECS间互访流量走向 5.对等连接流量走向&#xff08;跨V…

计算机网络:运输层 —— TCP 的拥塞控制

文章目录 TCP的拥塞控制拥塞控制的基本方法流量控制与拥塞控制的区别拥塞控制分类闭环拥塞控制算法 TCP的四种拥塞控制方法&#xff08;算法&#xff09;窗口慢开始门限慢开始算法拥塞避免算法快重传算法快恢复算法 TCP拥塞控制的流程TCP拥塞控制与网际层拥塞控制的关系 TCP的拥…

利用uniapp开发鸿蒙:运行到鸿蒙模拟器—踩坑合集

从uniapp运行到鸿蒙模拟器上这一步&#xff0c;就有非常多的坑&#xff0c;一些常见的坑&#xff0c;官网都有介绍&#xff0c;就不再拿出来了&#xff0c;这里记录一下官网未记录的大坑 1.运行路径从hbuilderx启动鸿蒙模拟器 解决方法&#xff1a; Windows系统&#xff0c;官…

linux 常用命令指南(存储分区、存储挂载、docker迁移)

前言&#xff1a;由于目前机器存储空间不够&#xff0c;所以‘斥巨资’加了一块2T的机械硬盘&#xff0c;下面是对linux扩容的一系列操作&#xff0c;包含了磁盘空间的创建、删除&#xff1b;存储挂载&#xff1b;docker迁移&#xff1b;anaconda3迁移等。 一、存储分区 1.1 …

layui合并table相同内的行

<table border"1" id"table1" class"layui-table"><thead><tr><th><b>姓名</b></th><th><b>项目</b></th><th><b>任务</b></th><th><b>…

C++刷题强训(day10)--最长回文子串、买股票的最好时期(一)、过河卒

目录 1、最长回文子串 1.1 题目 1.2 思路 1.3 代码实现 2、买卖股票的最好时机 2.1 题目 2.2 思路 2.3 代码实现 3、过河卒 3.1 题目 3.2 思路 3.3 代码实现 1、最长回文子串 1.1 题目 1.2 思路 根据题目可知&#xff0c;在一个长度为n的字符串中求得最长回文子…

【蓝桥杯C/C++】翻转游戏:多种实现与解法解析

博客主页&#xff1a; [小ᶻZ࿆] 本文专栏: 蓝桥杯C/C 文章目录 &#x1f4af;题目&#x1f4af;问题分析解法一&#xff1a;减法法解法二&#xff1a;位运算解法解法三&#xff1a;逻辑非解法解法四&#xff1a;条件运算符解法解法五&#xff1a;数组映射法不同解法的比较…

Debezium-BinaryLogClient

文章目录 概要核心流程技术名词解释技术细节小结 概要 BinaryLogClient类&#xff0c;用于连接和监听 MySQL 服务器的二进制日志&#xff08;binlog&#xff09; 核心流程 技术名词解释 ### GTID (Global Transaction Identifier) 理解 #### 定义 GTID&#xff08;Global Tra…

嵌入式linux中QT信号与槽基本操作与实现

大家好,今天主要给大家分享一下,如何使用linux系统上的QT进行界面开发与实现。 第一:QT的信号与槽基本简介 在操作QT的时候,可以使用里面的信号与槽。所谓信号就是一个对象发出的信号,槽就是当这个对象发出这个信号时,对应连接的槽就发被执行或者触发。 进行信号与槽的连…

03 —— Webpack 自动生成 html 文件

HtmlWebpackPlugin | webpack 中文文档 | webpack中文文档 | webpack中文网 安装 npm install --save-dev html-webpack-plugin 下载html-webpack-plugin本地软件包 npm i html-webpack-plugin --save-dev 配置webpack.config.js让webpack拥有插件功能 const HtmlWebpack…

大模型时代的具身智能系列专题(十二)

Robert Platt(波士顿动力) Robert Platt是美国东北大学Helping Hands机器人实验室主任、计算机科学教授。在加入东北大学之前&#xff0c;Platt 曾是麻省理工学院的研究科学家和美国宇航局的机器人工程师。platt博士毕业于马萨诸塞大学阿默斯特分校计算机科学专业。Platt 的工…

【软件测试】设计测试用例的万能公式

文章目录 概念设计测试用例的万能公式常规思考逆向思维发散性思维万能公式水杯测试弱网测试如何进行弱网测试 安装卸载测试 概念 什么是测试用例&#xff1f; 测试⽤例&#xff08;Test Case&#xff09;是为了实施测试⽽向被测试的系统提供的⼀组集合&#xff0c;这组集合包…

uni-app Vue3语法实现微信小程序样式穿透uview-plus框架

1 问题描述 我在用 uni-app vue3 语法开发微信小程序时&#xff0c;在项目中使用了 uview-plus 这一开源 UI 框架。在使用 up-text 组件时&#xff0c;想要给它添加一些样式&#xff0c;之前了解到微信小程序存在样式隔离的问题&#xff0c;也在uview-plus官网-注意事项中找到…

C++(Qt)软件调试---内存分析工具Heob(26)

C(Qt)软件调试—内存分析工具Heob&#xff08;26&#xff09; 文章目录 C(Qt)软件调试---内存分析工具Heob&#xff08;26&#xff09;[toc]1、概述&#x1f41c;2、环境配置&#x1fab2;3、功能说明4、使用Heob分析qt 程序内存泄漏&#x1f9a7;5、使用Heob检测qt 程序野指针…

uni-app快速入门(八)--常用内置组件(上)

uni-app提供了一套基础组件&#xff0c;类似HTML里的标签元素&#xff0c;不推荐在uni-app中使用使用div等HTML标签。在uni-app中&#xff0c;对应<div>的标签是view&#xff0c;对应<span>的是text&#xff0c;对应<a>的是navigator&#xff0c;常用uni-app…

静态时序分析--时序约束

目录 1.时钟约束1.1创建时钟1.2.生成时钟1.3虚拟时钟1.4 最小时钟脉宽 2.I/O延时约束2.1设置输入延时2.2设置输出延时 3.I/O环境建模约束3.1输入驱动建模3.2输出负载建模 4.时序例外4.1多周期路径设置&#xff08;multicycle path&#xff09;4.2伪路径设置&#xff08;false_p…