适配器模式

定义

​适配器模式(Adapter Pattern)是指将一个类的接口转换成客户期望的另一个接口,使原本的接口不兼容的类可以一起工作,属于结构型设计模式。


适用场景

1、已经存在的类,它的方法和需求不匹配(方法结果相同或相似)的情况。

2、适配器模式不是软件设计阶段考虑的设计模式,是随着软件维护,由于不同产品、不同厂家造成功能类似而接口不相同情况下的解决方案。有点亡羊补牢的感觉。

实例

我们现在的登录有很多种,QQ、微信等,我们利用适配模式来写下这个代码

创建统一的返回结果类

public class ResultMsg {private int code;private String msg;private Object data;public ResultMsg(int code, String msg, Object data) {this.code = code;this.msg = msg;this.data = data;    }public int getCode() {return code;    }public void setCode(int code) {this.code = code;    }public String getMsg() {return msg;    }public void setMsg(String msg) {this.msg = msg;    }public Object getData() {return data;    }public void setData(Object data) {this.data = data;    }}

我们假设原来的系统的登录逻辑是SinginService

public class SiginService {/**     * 注册方法     * @param username     * @param password     * @return     */public ResultMsg regist(String username,String password){return  new ResultMsg(200,"注册成功",new Member());    }/**     * 登录的方法     * @param username     * @param password     * @return     */public ResultMsg login(String username,String password){return null;    }}

创建Member类

public class Member {private String username;private String password;private String mid;private String info;public String getUsername() {return username;    }public void setUsername(String username) {this.username = username;    }public String getPassword() {return password;    }public void setPassword(String password) {this.password = password;    }public String getMid() {return mid;    }public void setMid(String mid) {this.mid = mid;    }public String getInfo() {return info;    }public void setInfo(String info) {this.info = info;    }}

原来的逻辑代码我们不动,新创建一个类继承

public class SinginForThirdService extends SiginService {public ResultMsg loginForQQ(String openId){//1、openId是全局唯一,我们可以把它当做是一个用户名(加长)//2、密码默认为QQ_EMPTY//3、注册(在原有系统里面创建一个用户)//4、调用原来的登录方法return loginForRegist(openId,null);    }public ResultMsg loginForWechat(String openId){return null;    }public ResultMsg loginForToken(String token){//通过token拿到用户信息,然后再重新登陆了一次return  null;    }public ResultMsg loginForTelphone(String telphone,String code){return null;    }public ResultMsg loginForRegist(String username,String password){super.regist(username,null);return super.login(username,null);    }}

客户端测试代码

public static void main(String[] args) {        SinginForThirdService service = new SinginForThirdService();        service.login("tom","123456");        service.loginForQQ("sdfasdfasf");        service.loginForWechat("sdfasfsa");    }

通过一个简单的适配,完成了代码的兼容​。当然我们的代码还可以优化,很多人可能想到了策略模式,工厂模式​。我们下面来优化下

根据不同的登录方式,创建不同的Adapter,首先创建LoginAdapter

public interface LoginAdapter {boolean support(Object adapter);    ResultMsg login(String id,Object adapter);}

分别实现不同的登录适配,QQ登录LoginForQQAdapter

public class LoginForQQAdapter implements LoginAdapter {public boolean support(Object adapter) {return adapter instanceof LoginForQQAdapter;    }public ResultMsg login(String id, Object adapter) {return null;    }}

同理,其他的登录也是一样,我直接上代码

public class LoginForSinaAdapter implements LoginAdapter {public boolean support(Object adapter) {return adapter instanceof LoginForSinaAdapter;    }public ResultMsg login(String id, Object adapter) {return null;    }}
public class LoginForTelAdapter implements LoginAdapter {public boolean support(Object adapter) {return adapter instanceof LoginForTelAdapter;    }public ResultMsg login(String id, Object adapter) {return null;    }}
public class LoginForTokenAdapter implements LoginAdapter {public boolean support(Object adapter) {return adapter instanceof LoginForTokenAdapter;    }public ResultMsg login(String id, Object adapter) {return null;    }}
public class LoginForWechatAdapter implements LoginAdapter {public boolean support(Object adapter) {return adapter instanceof LoginForWechatAdapter;    }public ResultMsg login(String id, Object adapter) {return null;    }}

然后创建第三方登录兼容接口

public interface IPassportForThird {/**     * QQ登录     * @param id     * @return     */ResultMsg loginForQQ(String id);/**     * 微信登录     * @param id     * @return     */ResultMsg loginForWechat(String id);/**     * 记住登录状态后自动登录     * @param token     * @return     */ResultMsg loginForToken(String token);/**     * 手机号登录     * @param telphone     * @param code     * @return     */ResultMsg loginForTelphone(String telphone, String code);/**     * 注册后自动登录     * @param username     * @param passport     * @return     */ResultMsg loginForRegist(String username, String passport);}

实现兼容类

public class PassportForThirdAdapter extends SiginService implements IPassportForThird {public ResultMsg loginForQQ(String id) {//        return processLogin(id,RegistForQQAdapter.class);return processLogin(id,LoginForQQAdapter.class);    }public ResultMsg loginForWechat(String id) {return processLogin(id,LoginForWechatAdapter.class);    }public ResultMsg loginForToken(String token) {return processLogin(token,LoginForTokenAdapter.class);    }public ResultMsg loginForTelphone(String telphone, String code) {return processLogin(telphone,LoginForTelAdapter.class);    }public ResultMsg loginForRegist(String username, String passport) {super.regist(username,passport);return super.login(username,passport);    }private ResultMsg processLogin(String key,Class extends LoginAdapter> clazz){try{//适配器不一定要实现接口            LoginAdapter adapter = clazz.newInstance();//判断传过来的适配器是否能处理指定的逻辑if(adapter.support(adapter)){return adapter.login(key,adapter);            }        }catch (Exception e){            e.printStackTrace();        }return null;    }}

客户端测试代码

public class PassportTest {public static void main(String[] args) {        IPassportForThird passportForThird = new PassportForThirdAdapter();        passportForThird.loginForQQ("");    }}

看下类图

优点

1、能提高类的透明性和复用,现有的类复用但不需要改变。

2、目标类和适配器类解耦,提高程序的扩展性。

3、在很多业务场景中符合开闭原则。

缺点

1、适配器编写过程需要全面考虑,可能会增加系统的复杂性。

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

对接多种三方的设计模式_死磕设计模式之适配器模式相关推荐

  1. 3 v4 中心节点固定_死磕以太坊源码分析之p2p节点发现

    死磕以太坊源码分析之p2p节点发现 在阅读节点发现源码之前必须要理解kadmilia算法,可以参考:KAD算法详解. 节点发现概述 节点发现,使本地节点得知其他节点的信息,进而加入到p2p网络中. 以 ...

  2. yield方法释放锁吗_死磕Synchronized底层实现重量级锁

    点击上方"Java知音",选择"置顶公众号" 技术文章第一时间送达! 作者:farmerjohngit 链接:https://github.com/farmer ...

  3. activity 生命周期_死磕Android_App 启动过程(含 Activity 启动过程)

    1. 前言 Activity是日常开发中最常用的组件,系统给我们做了很多很多的封装,让我们平时用起来特别简单,很顺畅.但是你有没有想过,系统内部是如何启动一个Activity的呢?Activity对象 ...

  4. java 同步锁_死磕 java同步系列之自己动手写一个锁Lock

    问题 (1)自己动手写一个锁需要哪些知识? (2)自己动手写一个锁到底有多简单? (3)自己能不能写出来一个完美的锁? 简介 本篇文章的目标一是自己动手写一个锁,这个锁的功能很简单,能进行正常的加锁. ...

  5. tomcat中间件的默认端口号_死磕Tomcat系列(1)——整体架构

    点击上方"Java技术前线",选择"置顶或者星标" 与你一起成长 在许多的高端开发的岗位中都会或多或少有要求面试人员要研究过一些常用中间件源码.这是因为一切的秘 ...

  6. java任务流程_死磕 java线程系列之线程池深入解析——普通任务执行流程

    (手机横屏看源码更方便) 注:java源码分析部分如无特殊说明均基于 java8 版本. 注:线程池源码部分如无特殊说明均指ThreadPoolExecutor类. 简介 前面我们一起学习了Java中 ...

  7. java 外部接口调用 设计模式_《Java设计模式》之接口模式

    -----------模式是思想的体现,而非具体的实现. 抽象的讲,类的接口是类允许其他类对象访问的方法与字段集.接口通常代表一种承诺,即方法需要实现接口方法名表示的操作,遵循代码注释和其他文档说明, ...

  8. java线程池深入讲解_死磕 java线程系列之线程池深入解析——生命周期

    (手机横屏看源码更方便) 注:java源码分析部分如无特殊说明均基于 java8 版本. 注:线程池源码部分如无特殊说明均指ThreadPoolExecutor类. 简介 上一章我们一起重温了下线程的 ...

  9. java 手编线程池_死磕 java线程系列之自己动手写一个线程池

    欢迎关注我的公众号"彤哥读源码",查看更多源码系列文章, 与彤哥一起畅游源码的海洋. (手机横屏看源码更方便) 问题 (1)自己动手写一个线程池需要考虑哪些因素? (2)自己动手写 ...

最新文章

  1. redis 内存溢出_Redis为何这么快数据存储角度
  2. SQL Server 2008 开启远程连接
  3. 21朵水仙花算法java,柏拉图说,如果你有两块面包,你应当用一块去换一朵水仙花。。。。。...
  4. 4 年 46 个版本,一文读懂 Spring Cloud 发展历史
  5. OpenJ_Bailian 4017 爬楼梯
  6. 浏览器“四巨头”首度合作 解决网页适配问题
  7. python学习笔记-Day5
  8. Cesium:向地图中添加线的方法
  9. Amesim(七):amesim自带demo动力学模型研究
  10. 性能监控之 Linux 命令 top、vmstat、iostat、free、iftop 基础
  11. CVPR 2022 NTIRE 高动态范围成像(HDR)比赛冠军方案
  12. 在ie6下实现position-fixed的效果
  13. uniapp 消息提示框
  14. 华夏千秋事,风雨亦为之
  15. 趣图:看到网友晒了新抱枕,我也想换个新的了
  16. 差分放大电路的基本工作原理是什么//2021-2-18
  17. Bingo说说:如何选对赚钱项目?3招教你不再迷茫
  18. vscode 实用插件
  19. js日期字符串(年月日)与时间戳的转换
  20. 智慧小区三维可视化综合管理平台

热门文章

  1. 一个假设:如果两个量互为分类对象和分类载体则他们不可对易
  2. 卷积核一定可以提升网络性能吗?-分类0,2
  3. sharp扫地机器人讲话_扫地机机器人,智能扫地机器人推荐
  4. P11 非线性系统-《Matlab/Simulink与控制系统仿真》程序指令总结
  5. 【PC工具】大神开源项目-配色调色工具Colorpicker颜色选择器
  6. 国内勒索病毒疫情严重 每天十多万台电脑被感染
  7. 【FTP】java FTPClient 文件上传内容为空,文件大小为0
  8. 内存溢出和内存泄漏的区别、产生原因以及解决方案 转
  9. 高性能Cordova App开发学习笔记
  10. Android实例-手机震动(XE8+小米2)