首页 > 其他分享 >一文彻底带你搞懂什么是适配器模式!!

一文彻底带你搞懂什么是适配器模式!!

时间:2024-07-08 21:27:51浏览次数:16  
标签:一文 适配 适配器 接口 110V 充电 搞懂 Adaptee

一文彻底带你搞懂什么是适配器模式!!

什么是适配器模式?

适配器模式(Adapter Pattern)是作为两个不兼容的接口之间的桥梁。这种类型的设计模式属于结构型模式,它结合了两个独立接口的功能。

主要目的是解决在不改变现有代码的情况下,使不兼容的接口之间能够正常工作,通过创建一个中间转换的适配器来将一个对象转换成我们所需要的接口

例如下图的 USB 转 Type-C 就是一个适配器
在这里插入图片描述

适配器的两种实现方式

通常有两种方式实现适配器模式,一种是类适配器,类适配器目前已不太使用,另一种实现方式是对象适配器,通常情况下采用对象适配器会使得代码更易扩展与维护。

不管采用何种方式,其基本的实现思想都是:对现有接口的实现类进行扩展,使其实现客户期望的目标接口。

类适配器通过继承现有接口类并实现目标接口,这样的话会使得现有接口类完全对适配器暴露,使得适配器具有现有接口类的全部功能,破坏了封装性。此外从逻辑上来说,这也是不符合常理的,适配器要做的是扩展现有接口类的功能而不是替代。

对象适配器持有现有接口类一个实例,并扩展其功能,实现目标接口。这是推荐的方式,优先采用组合而不是继承,会使得代码更利于维护。

适用情况

  1. 接口不兼容:你有两个接口或类,它们原本不兼容,但你又想让它们能够一起工作。

  2. 希望复用代码:你已经有一个功能完整的类或接口,但接口不符合新的需求。通过适配器,你可以重用现有的代码,而不是重新实现一个新的版本。

  3. 第三方集成:你可能需要使用第三方库或类,但第三方库的接口与你自己的接口不兼容。使用适配器可以轻松地将第三方库集成到你的系统中。

代码示例

背景

美国的正常供电电压为110V,一个中国人带了一款中国制造电器去美国,这个电器必须要在220V电压下才能充电使用。

这种情况下,客户(中国人)的期望接口是有一个220V的电压为电器充电,但实际的接口是仅有一个110V的电压供电器充电,所以就需要采用一根电压转换器(适配器)使得110V的电压能够转换为220V的电压,供客户使用。

  • Target:客户期望获得的功能接口(220V电压供电)。
  • Cilent:客户,期望访问Target接口(客户期望能有220V电压)。
  • Adaptee:现有接口,这个接口需要被适配(现有110V电压供电,需要被适配至220V)。
  • Adapter:适配器类,适配现有接口使其符合客户需求接口(适配110V电压,使其变为220V电压)。

在适配器模式中,Adapter扩展Adaptee以实现对应功能,Cilent调用Adapter以获得相应功能。

类适配器

类适配器结构图:

在这里插入图片描述

详细附代码图:

在这里插入图片描述

// 定义一个目标接口,它声明了期望实现的充电方法chargeBy220V
interface Target {
    void chargeBy220V();
}

// 定义一个被适配的类接口,它有chargeBy110V充电方法
interface Adaptee {
    void chargeBy110V();
}

// 创建一个美国供电器类,它实现了被适配的类接口,提供110V充电方法
class AmericanCharger implements Adaptee {
    @Override
    public void chargeBy110V() {
        System.out.println("美国供电器,为您服务,正在给您充 110V 电压");
    }
}

// 创建一个适配器类,它继承了美国供电器类,并且实现了目标接口
class Adapter extends AmericanCharger implements Target {
    // 适配器重写目标接口的chargeBy220V方法,
    // 调用被适配类的chargeBy110V方法,并额外转换电压
    @Override
    public void chargeBy220V() {
        super.chargeBy110V(); // 调用被适配类的110V充电方法
        System.out.println("再加 110V,现在给您充 220V 电压!!"); // 实现额外的电压转换逻辑
    }
}

// 测试类,测试适配器模式
public class Client{
    public static void main(String[] args) {
        // 创建适配器对象,并通过目标接口调用chargeBy220V方法
        new Adapter().chargeBy220V();
    }
}

在这里插入图片描述

这种类适配器可能会让人误以为适配器本身就是提供110V充电功能的美国供电器,而忽略了它实际上是一个转换器,逻辑上容易混乱和让人混淆

对象适配器

在这里插入图片描述

在这里插入图片描述

/**
 * 定义一个目标接口Target,包含一个充电方法chargeBy220V,该方法用于充电220V电压。
 */
interface Target {
    void chargeBy220V(); // 定义接口方法,使用220V电压充电
}

/**
 * 定义一个适配者接口Adaptee,包含一个充电方法chargeBy110V,该方法用于充电110V电压。
 */
interface Adaptee {
    void chargeBy110V(); // 定义接口方法,使用110V电压充电
}

/**
 * 实现适配者接口Adaptee的AmericanCharger类,代表美国的充电器,提供110V的充电服务。
 */
class AmericanCharger implements Adaptee {
    @Override
    public void chargeBy110V() { // 覆写chargeBy110V方法,输出充电信息
        System.out.println("美国供电器,为您服务,正在给您充 110V 电压");
    }
}

/**
 * 实现目标接口Target的Adapter类,通过持有Adaptee对象来适配不同的电压充电需求。
 */
class Adapter implements Target {
    private Adaptee adaptee; // 私有变量,用于持有Adaptee类型的对象

    /**
     * 构造器,接收一个Adaptee对象以便在内部调用其方法。
     * @param adaptee 适配者对象
     */
    public Adapter(Adaptee adaptee) { // 构造器初始化适配器,并传入适配者对象
        this.adaptee = adaptee;
    }

    @Override
    public void chargeBy220V() { // 覆写chargeBy220V方法,实现适配功能
        adaptee.chargeBy110V(); // 首先使用110V充电
        System.out.println("再加 110V,现在给您充 220V 电压!!"); // 然后输出适配后的信息
    }
}

/**
 * 测试类Test,用于演示对象适配模式的使用。
 */
public class Client{
    public static void main(String[] args) {
    
        Adaptee adaptee = new AmericanCharger(); // 创建一个AmericanCharger对象

        Adapter adapter = new Adapter(adaptee); // 使用AmericanCharger对象来创建Adapter对象

        adapter.chargeBy220V(); // 通过Adapter对象调用chargeBy220V方法,输出适配后的充电信息
    }
}

在这里插入图片描述

对象适配器采用组合的方式实现对现有接口的扩展以达到客户期望的接口。

IO流中的实际应用

让我们来看JavaIO流中的一个实例:

FileInputStream fis = new FileInputStream("qe");
InputStreamReader isrAdapter = new InputStreamReader(fis);
BufferedReader bf = new BufferedReader(isrAdapter);

BufferedReader (此处也就是客户)需要读取文件字符流进行工作,读取文件字符流就是客户的需求部分,但是根据现有的接口,想要读取文件就只能读取字节流

FileInputStream 就是现有接口的一个具体实现类,为了满足客户的需求,我们要对现有的接口进行适配,

InputStreamReader 就是一个适配器,它持有一个现有接口类的实例,通过这个实例读取文件字节流并将其扩展为字符流以满足客户的需求,这是标准的对象适配器模式。

如果仔细研究源码,发现JavaIO库将适配器定义为抽象的,并由具体的适配器继承该抽象适配器,如这里的 InputStreamReader 就是具体的适配器之一。

应用扩展

如果实现适配有多种方式的话,我们可以将适配器类Adapter声明为抽象类,并由子类扩展它:

在这里插入图片描述

在这里插入图片描述

/**
 * 定义一个目标接口Target,
 * 包含一个充电方法chargeBy220V,该方法用于充电220V电压。
 */
interface Target {
    void chargeBy220V(); // 目标接口的充电方法,使用220V电压充电
}

/**
 * 定义一个适配者接口Adaptee,
 * 包含一个充电方法chargeBy110V,该方法用于充电110V电压。
 */
interface Adaptee {
    void chargeBy110V(); // 适配者接口的充电方法,使用110V电压充电
}

/**
 * 实现适配者接口Adaptee的AmericanCharger类,
 * 代表美国的充电器,提供110V的充电服务。
 */
class AmericanCharger implements Adaptee {
    @Override
    public void chargeBy110V() {
        System.out.println("美国供电器,为您服务,正在给您充 110V 电压");
    }
}

/**
 * 抽象类Adapter,作为适配器的基础类,实现目标接口Target。
 */
abstract class Adapter implements Target {
    Adaptee adaptee; // 私有变量,用于持有Adaptee类型的对象

    /**
     * 构造器,接收一个Adaptee对象以便在内部调用其方法。
     * @param adaptee 适配者对象
     */
    public Adapter(Adaptee adaptee) { // 构造器初始化适配器,并传入适配者对象
        this.adaptee = adaptee;
    }
}

/**
 * 实现Adapter类的具体子类ChinaMakeAdapter,
 * 用于将美国充电器的110V充电适配为中国标准的220V充电。
 */
class ChinaMakeAdapter extends Adapter{

    public ChinaMakeAdapter(Adaptee adaptee){
        super(adaptee); // 调用父类的构造器
    }

    /**
     * 实现目标接口Target的chargeBy220V方法,
     * 通过调用适配者接口Adaptee的chargeBy110V方法来实现。
     */
    @Override
    public void chargeBy220V() {
        adaptee.chargeBy110V(); // 调用适配者的110V充电方法
        System.out.println("再加110V,达到220V,认准中国制造!"); // 输出说明适配器正在工作
    }
}

/**
 * 测试类Client,用于演示对象适配器模式的使用。
 */
public class Client{
    public static void main(String[] args) {
    	// 创建一个AmericanCharger对象
        Adaptee adaptee = new AmericanCharger(); 
         // 使用AmericanCharger对象来创建ChinaMakeAdapter对象
        Adapter adapter = new ChinaMakeAdapter(adaptee);
        // 通过ChinaMakeAdapter对象调用chargeBy220V方法,输出适配后的充电信息
        adapter.chargeBy220V(); 
    }
}


在这里插入图片描述

总结

优点:

  1. 解耦合:适配器模式允许不相关的类或接口协同工作,这为系统的不同部分提供了更大的灵活性。
  2. 复用:通过适配器,可以复用现有的类,而不需要修改它们的源代码。
  3. 透明性:适配器提供了一种封装机制,允许客户端代码与目标接口交互,而不必关心实际的实现类。
  4. 灵活性:适配器模式提供了很大的灵活性,因为它允许在运行时动态地结合不同的类和接口。

缺点:

  1. 过多地使用适配器,会让系统非常零乱,不易整体进行把握。比如,明明看到调用的是 A 接口,其实内部被适配成了 B 接口的实现,一个系统如果太多出现这种情况,无异于一场灾难。因此如果不是很有必要,可以不使用适配器,而是直接对系统进行重构。
  2. 适配可能复杂或不可能:有时,适配两个不兼容的接口可能非常困难,甚至不可能,比如让母猪飞上天。

当我们有动机地修改一个正常运行的系统的接口,这时应该考虑使用适配器模式。

注意事项:适配器不是在详细设计时添加的,而是解决正在服役的项目的问题,即现有接口可能无法改变(去美国不可能把人家110V电压供给改成220V电压供给)。

往期设计模式专题文:

一文带你彻底搞懂什么是代理模式!!

一文带你彻底搞懂设计模式之单例模式!!由浅入深,图文并茂,超超超详细的单例模式讲解!!

标签:一文,适配,适配器,接口,110V,充电,搞懂,Adaptee
From: https://blog.csdn.net/m0_63653444/article/details/140261705

相关文章

  • 【汽车故障诊断4】一文了解故障诊断码DTC验证程度、快照和扩展数据
    在上篇【汽车故障诊断3】一文了解诊断故障码DTC状态位介绍DTC状态位,本篇文章将继续介绍DTC的其他信息:DTC严重程度,DTC快照和DTC扩展数据。1什么是DTC严重程度  DTC严重程度占用1个字节数据,包含两部分信息,DTC严重程度信息(3位)和DTC类别信息(5位),如下所示:source:ISO15031-6 ......
  • 什么是静态住宅代理?一文看懂它
    静态住宅代理(也称为ISP代理)是最流行的代理类型之一。它们也是隐藏身份和在线匿名的最佳方式之一。但是您需要了解它们什么?是什么让它们如此特别?为什么您要使用住宅代理而不是仅仅使用常规代理服务?如果你感兴趣,那么就看下去!什么是静态住宅代理?首先,我们来谈谈住宅代理。住宅......
  • Python排序,你用对了吗?一文教你sorted和sort的正确姿势!
    目录1、sorted基础用法......
  • 一文带你了解什么是工控机?
    ​在工业计算领域,无风扇系统因其独特的设计和众多优势而获得了巨大的关注。与依靠风扇进行冷却的传统计算机不同,无风扇工业计算机经过精心设计,无需移动部件即可散热。在这篇文章中,我们将探讨无风扇工业计算机的引人注目的优势以及它们在各个行业中越来越受欢迎的原因。增强的可靠......
  • 【Sklearn-驯化】一文学会机器学习中的交叉验证-Kflods使用技巧
    【Sklearn-驯化】一文学会机器学习中的交叉验证-Kflods使用技巧 本次修炼方法请往下查看......
  • 一文带你彻底搞懂什么是代理模式!!
    一文带你彻底搞懂什么是代理模式!!什么是代理模式?静态代理静态代理的问题动态代理JDK动态代理JDK动态代理的问题CGLIB动态代理JDK动态代理和CGLIB动态代理对比静态代理和动态代理的对比总结什么是代理模式?代理模式是一种常见的设计模式,它为其他对象提供了一种......
  • STL复习-序列式容器和容器适配器部分
    STL复习1.常见的容器如何介绍这些容器,分别从常见接口,迭代器类型,底层实现序列式容器stringstring严格来说不属于stl,它是属于C++标准库**底层实现:**string本质是char类型的顺序表,因为不同编译器下的具体实现不同,这里只提供一个我的简答框架classstring{public:......
  • 一文带你了解Java虚拟机垃圾收集器
    如果你的世界,没有痛苦的害怕,没有尊严的担忧,没有富贵的贫贱,没有暖寒的交替,没有外貌的困扰,没有男女的区别,没有你我之分,没有生死顾虑,你才会离"真正的活着"越来越近。——余华《活着》在上一篇内容中我们提到在Java虚拟机中主要是通过标记整理、复制拷贝、标记清除三种算法进行内......
  • 一文讲透箭头函数和This
    目录前言环境说明准备node中this的坑this常用场景全局中的this函数中的this构造函数中的this改变this的三种方式箭头函数箭头函数This结论代码示例面试题测试:前言在JavaScript中,this是理解函数上下文的关键。本文将深入剖析this的奥秘,揭示其在不同场景下的行为,......
  • 一文理解 Treelite,Treelite 为决策树集成模型的部署和推理提供了高效、灵活的解决方案
    ......