Java设计模式-适配器模式 理论代码相结合

简介: Java设计模式-适配器模式 理论代码相结合

微信截图_20220524185217.png

继代理模式后又来到适配器模式啦,想看之前的也有哦。持续更新中哦。让我们一起加油吧兄弟们,干他。


很喜欢一句话:”八小时内谋生活,八小时外谋发展".

你好,如果喜欢,请一起坚持!!

共勉😁


一张旧照,恍惚间想起旧人


设计模式系列


一、前言


1)概述


在现实生活中,经常出现两个对象因接口不兼容而不能在一起工作的实例,这时需要第三者进行适配。例如,讲中文的人同讲英文的人对话时需要一个翻译,用直流电的笔记本电脑接交流电源时需要一个电源适配器,用计算机访问照相机的 SD 内存卡时需要一个读卡器等。还有像下面这张图一样:


微信截图_20220524185332.png


在软件设计中也可能出现:需要开发的具有某种业务功能的组件在现有的组件库中已经存在,但它们与当前系统的接口规范不兼容,如果重新开发这些组件成本又很高,这时用适配器模式能很好地解决这些问题。


2)介绍


适配器模式(Adapter)的定义如下:将一个类的接口转换成客户希望的另外一个接口,使得原本由于接口不兼容而不能一起工作的那些类能一起工作。适配器模式分为类结构型模式和对象结构型模式两种,前者类之间的耦合度比后者高,且要求程序员了解现有组件库中的相关组件的内部结构,所以应用相对较少些。


Adapter模式的宗旨:保留现有类所提供的服务,向客户提供接口,以满足客户的期望


3)角色结构


适配器模式(Adapter)包含以下主要角色:


  • 目标(Target)接口:当前系统业务所期待的接口,它可以是抽象类或接口。


  • 适配者(Adaptee)类:它是被访问和适配的现存组件库中的组件接口。


  • 适配器(Adapter)类:它是一个转换器,通过继承或引用适配者的对象,把适配者接口转换成目标接口,让客户按目标接口的格式访问适配者。


4)使用场景


适配器模式(Adapter)通常适用于以下场景。


  • 以前开发的系统存在满足新系统功能需求的类,但其接口同新系统的接口不一致。(就是所谓的加一层,一层不行就加两层)😁


  • 使用第三方提供的组件,但组件接口定义和自己要求的接口定义不同。


二、类适配器


当客户在接口中定义了他期望的行为时,我们就可以应用适配器模式,提供一个实现该接口的类,并且扩展已有的类,通过创建子类来实现适配。


实现方式定义一个适配器类来实现当前系统的业务接口,同时又继承现有组件库中已经存在的组件


我们直接用之前的那个图来做个例子:中国人到了欧洲,的给自己电脑充电,但因为自己电脑是双叉,欧式是三叉,这中间就得需要一个转换器。


微信截图_20220524185530.png


2.1、代码


目标(Target)接口:即图中的欧式三叉


public interface EuropeSocket {
    /** 欧式三叉 通电 接通电 插座*/
    String useEuropesocket();
}
// 欧式三叉实现类
public class EuropeSocketImpl implements EuropeSocket {
    @Override
    public String useEuropesocket() {
        String msg ="使用欧式三叉充电";
        return msg;
    }
}


适配者(Adaptee):即中国双叉


public interface ChineseSocket {
    /**
     * 使用中国双叉充电
     * @return
     */
    String useChineseSocket();
}
// 中国插头的实现类
public class ChineseSocketImpl implements ChineseSocket {
    @Override
    public String useChineseSocket() {
        String msg="使用中国双叉充电";
        return msg;
    }
}


适配器(Adapter)类:


/**
 * 定义适配器类 中国双叉转为欧洲三叉
 *
 */
public class ChineseAdapterEurope extends EuropeSocketImpl implements ChineseSocket {
    @Override
    public String useChineseSocket() {
        System.out.println("使用转换器转换完成");
        return useEuropesocket();
    }
}


电脑类


public class Computer {
    public String useChineseSocket(ChineseSocket chineseSocket) {
        if(chineseSocket == null) {
            throw new NullPointerException("sd card null");
        }
        return chineseSocket.useChineseSocket();
    }
}


测试:


public class Client {
    public static void main(String[] args) {
        Computer computer = new Computer();
        ChineseSocket chineseSocket = new ChineseSocketImpl();
        System.out.println(computer.useChineseSocket(chineseSocket));
        System.out.println("------------");
        ChineseAdapterEurope adapter = new ChineseAdapterEurope();
        System.out.println(computer.useChineseSocket(adapter));
        /**
         * 输出:
         * 使用中国双叉充电
         * ------------
         * 使用转换器转换完成
         * 使用欧式三叉充电
         */
    }
}


上述代码就是简单的演示了适配器的使用。


:类适配器模式违背了合成复用原则。类适配器是客户类有一个接口规范的情况下可用,反之不可用。


三、对象适配器


对象适配器”通过组合除了满足“用户期待接口”还降低了代码间的不良耦合。在工作中推荐使用“对象适配”。


实现方式:对象适配器模式可釆用将现有组件库中已经实现的组件引入适配器类中,该类同时实现当前系统的业务接口


题目还是和上面一样的哈。代码其实差异很小


代码


目标(Target)接口:即图中的欧式三叉


public interface EuropeSocket {
    /** 欧式三叉 通电 接通电 插座*/
    String useEuropesocket();
}
// 欧式三叉实现类
public class EuropeSocketImpl implements EuropeSocket {
    @Override
    public String useEuropesocket() {
        String msg ="使用欧式三叉充电";
        return msg;
    }
}


适配者(Adaptee):即中国双叉


public interface ChineseSocket {
    /**
     * 使用中国双叉充电
     * @return
     */
    String useChineseSocket();
}
// 中国插头的实现类
public class ChineseSocketImpl implements ChineseSocket {
    @Override
    public String useChineseSocket() {
        String msg="使用中国双叉充电";
        return msg;
    }
}


适配器(Adapter)类: 就是这个适配器内做了一些更改 从继承改为了成员变量的方式


public class ChineseAdapterEurope implements ChineseSocket {
    private EuropeSocket europeSocket;
    public ChineseAdapterEurope(EuropeSocket europeSocket) {
        this.europeSocket = europeSocket;
    }
    @Override
    public String useChineseSocket() {
        System.out.println("使用转换器转换完成");
        return europeSocket.useEuropesocket();
    }
}


电脑类


public class Computer {
    public String useChineseSocket(ChineseSocket chineseSocket) {
        if(chineseSocket == null) {
            throw new NullPointerException("sd card null");
        }
        return chineseSocket.useChineseSocket();
    }
}


测试:


public class Client {
    public static void main(String[] args) {
        Computer computer = new Computer();
        ChineseSocket chineseSocket = new ChineseSocketImpl();
        System.out.println(computer.useChineseSocket(chineseSocket));
        System.out.println("------------");
        //这里做了更改
        EuropeSocket europeSocket=new EuropeSocketImpl();
        ChineseAdapterEurope adapter = new ChineseAdapterEurope(europeSocket);
        System.out.println(computer.useChineseSocket(adapter));
        /**
         * 输出:
         * 使用中国双叉充电
         * ------------
         * 使用转换器转换完成
         * 使用欧式三叉充电
         */
    }
}


这就是对象适配器啦,


适合于解决问题常见:


  •  需要的东西有,但不能用,且短时间无法改造。即,使得一个功能适合不同的环境。


  •   在开发中,系统的数据、行为都匹配,但接口不符时,可以考虑适配器。


  •   希望复用一些现存的类,但是接口又与复用环境的要求不一致,应该考虑用适配器模式。(使用一个已经存在的类,但它的接口(即,方法),与需要的不相同时)


扩展


适配器模式(Adapter)可扩展为双向适配器模式,双向适配器类既可以把适配者接口转换成目标接口,也可以把目标接口转换成适配者接口


四、总结


优点


  • 客户端通过适配器可以透明地调用目标接口。


  • 复用了现存的类,程序员不需要修改原有代码而重用现有的适配者类。提高了类的复用


  • 将目标类和适配者类解耦,解决了目标类和适配者类接口不一致的问题。灵活性好


  • 可以让任何两个没有关联的类一起运行


  • 在很多业务场景中符合开闭原则


其缺点是


  • 适配器编写过程需要结合业务场景全面考虑,可能会增加系统的复杂性。


  • 增加代码阅读难度,降低代码可读性,过多使用适配器会使系统代码变得凌乱。


  • (如:明明看到调用的是 A 接口,其实内部被适配成了 B 接口的实现,一个系统如果太多出现这种情况,无异于一场灾难)


五、自言自语


你卷我卷,大家卷,什么时候这条路才是个头啊。😇(还是直接上天吧)


有时候也想停下来歇一歇,一直做一个事情,感觉挺难坚持的。😁


你好,如果你正巧看到这篇文章,并且觉得对你有益的话,就给个赞吧,让我感受一下分享的喜悦吧,蟹蟹。🤗


如若有写的有误的地方,也请大家不啬赐教!!


同样如若有存在疑惑的地方,请留言或私信,定会在第一时间回复你。


持续更新中


目录
相关文章
|
2月前
|
Java 开发工具
【Azure Storage Account】Java Code访问Storage Account File Share的上传和下载代码示例
本文介绍如何使用Java通过azure-storage-file-share SDK实现Azure文件共享的上传下载。包含依赖引入、客户端创建及完整示例代码,助你快速集成Azure File Share功能。
369 5
|
2月前
|
Java 数据处理 API
为什么你的Java代码应该多用Stream?从循环到声明式的思维转变
为什么你的Java代码应该多用Stream?从循环到声明式的思维转变
261 115
|
2月前
|
安全 Java 编译器
为什么你的Java代码需要泛型?类型安全的艺术
为什么你的Java代码需要泛型?类型安全的艺术
186 98
|
2月前
|
安全 Java 容器
告别空指针噩梦:Optional让Java代码更优雅
告别空指针噩梦:Optional让Java代码更优雅
386 94
|
2月前
|
Java 编译器 API
java最新版和java8的区别,用代码展示
java最新版和java8的区别,用代码展示
276 43
|
设计模式 缓存 安全
Java设计模式的单例模式应用场景
Java设计模式的单例模式应用场景
286 4
|
设计模式 安全 Java
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
【JAVA】Java 中什么叫单例设计模式?请用 Java 写出线程安全的单例模式
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
207 4
|
设计模式 安全 Java
Java 编程中的设计模式:单例模式的深度解析
【9月更文挑战第22天】在Java的世界里,单例模式就像是一位老练的舞者,轻盈地穿梭在对象创建的舞台上。它确保了一个类仅有一个实例,并提供全局访问点。这不仅仅是代码优雅的体现,更是资源管理的高手。我们将一起探索单例模式的奥秘,从基础实现到高级应用,再到它与现代Java版本的舞蹈,让我们揭开单例模式的面纱,一探究竟。
117 11
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)

热门文章

最新文章