面试题:如何快速拿下第三方登录且易扩展?

简介: 我们使用适配模式来实现一个实际的业务场景,解决实际问题。年纪稍微大一点的小伙伴一定经历过这样的过程。很早以前开发的老系统应该都有登录接口,但是随着业务的发展和社会的进步,单纯地依赖用户名密码登录显然不能满足用户需求。
本文节选自《设计模式就该这样学》

1 使用类适配器重构第三方登录自由适配

我们使用适配模式来实现一个实际的业务场景,解决实际问题。年纪稍微大一点的小伙伴一定经历过这样的过程。很早以前开发的老系统应该都有登录接口,但是随着业务的发展和社会的进步,单纯地依赖用户名密码登录显然不能满足用户需求。现在,大部分系统都已经支持多种登录方式,如QQ登录、微信登录、手机登录、微博登录等,同时保留用户名密码的登录方式。虽然登录形式丰富,但是登录后的处理逻辑可以不必改,都是将登录状态保存到Session,遵循开闭原则。首先创建统一的返回结果ResultMsg类。

/**



 * Created by Tom.



 */



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;



    }



}

假设在老系统中,处理登录逻辑的代码在PassportService类中。

public class PassportService {



 



    /**



     * 注册方法



     * @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类。

/**



 * Created by Tom.



 */



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;



    }



}

我们也不改动运行非常稳定的代码,创建Target角色IPassportForThird接口。

public interface IPassportForThird {



 



    ResultMsg loginForQQ(String openId);



 



    ResultMsg loginForWechat(String openId);



 



    ResultMsg loginForToken(String token);



 



    ResultMsg loginForTelphone(String phone,String code);



 



}

创建Adapter角色实现兼容,创建一个新的类PassportForThirdAdapter,继承原来的逻辑。

public class PassportForThirdAdapter extends PassportService implements IPassportForThird {



 



    public ResultMsg loginForQQ(String openId) {



        return loginForRegist(openId,null);



    }



 



    public ResultMsg loginForWechat(String openId) {



        return loginForRegist(openId,null);



    }



 



    public ResultMsg loginForToken(String token) {



        return loginForRegist(token,null);



    }



 



    public ResultMsg loginForTelphone(String phone, String code) {



        return loginForRegist(phone,null);



    }



 



    private ResultMsg loginForRegist(String username,String password){



        if(null == password){



            password = "THIRD_EMPTY";



        }



        super.regist(username,password);



        return super.login(username,password);



    }



}

客户端测试代码如下。

public static void main(String[] args) {



        PassportForThirdAdapter adapter = new PassportForThirdAdapter();



        adapter.login("tom","123456");



        adapter.loginForQQ("sjooguwoersdfjhasjfsa");



        adapter.loginForWechat("slfsjoljsdo8234ssdfs");



}

2 使用接口适配器优化代码

通过这么一个简单的适配动作,我们完成了代码兼容。当然,代码还可以更加优雅,根据不同的登录方式,创建不同的Adapter。首先创建LoginAdapter接口。

public interface ILoginAdapter {



    boolean support(Object object);



    ResultMsg login(String id,Object adapter);



}

然后创建一个抽象类AbstractAdapter继承PassportService原有的功能,同时实现ILoginAdapter接口,再分别实现不同的登录适配,QQ登录LoginForQQAdapter。

public class LoginForQQAdapter extends AbstractAdapter{



    public boolean support(Object adapter) {



        return adapter instanceof LoginForQQAdapter;



    }



 



    public ResultMsg login(String id, Object adapter) {



        if(!support(adapter)){return null;}



        //accesseToken



        //time



        return super.loginForRegist(id,null);



 



    }



 



}

手机登录LoginForTelAdapter。

public class LoginForTelAdapter extends AbstractAdapter{



    public boolean support(Object adapter) {



        return adapter instanceof LoginForTelAdapter;



    }



 



    public ResultMsg login(String id, Object adapter) {



        return super.loginForRegist(id,null);



    }



}

Token自动登录LoginForTokenAdapter。

public class LoginForTokenAdapter extends AbstractAdapter {



    public boolean support(Object adapter) {



        return adapter instanceof LoginForTokenAdapter;



    }



 



    public ResultMsg login(String id, Object adapter) {



        return super.loginForRegist(id,null);



    }



}

微信登录LoginForWechatAdapter。

public class LoginForWechatAdapter extends AbstractAdapter{



    public boolean support(Object adapter) {



        return adapter instanceof LoginForWechatAdapter;



    }



 



    public ResultMsg login(String id, Object adapter) {



        return super.loginForRegist(id,null);



    }



}

接着创建适配器PassportForThirdAdapter类,实现目标接口IPassportForThird完成兼容。

public class PassportForThirdAdapter implements IPassportForThird {



 



    public ResultMsg loginForQQ(String openId) {



        return processLogin(openId, LoginForQQAdapter.class);



    }



 



    public ResultMsg loginForWechat(String openId) {



 



        return processLogin(openId, LoginForWechatAdapter.class);



 



    }



 



    public ResultMsg loginForToken(String token) {



 



        return processLogin(token, LoginForTokenAdapter.class);



    }



 



    public ResultMsg loginForTelphone(String phone, String code) {



        return processLogin(phone, LoginForTelAdapter.class);



    }



 



 



    private ResultMsg processLogin(String id,Class<? extends ILoginAdapter> clazz){



        try {



            ILoginAdapter adapter = clazz.newInstance();



            if (adapter.support(adapter)){



                return adapter.login(id,adapter);



            }



        } catch (Exception e) {



            e.printStackTrace();



        }



        return null;



    }



 



}

客户端测试代码如下。

public static void main(String[] args) {



        IPassportForThird adapter = new PassportForThirdAdapter();



        adapter.loginForQQ("sdfasdfasfasfas");



}

最后来看如下图所示的类图。

图片

至此,在遵循开闭原则的前提下,我们完整地实现了一个兼容多平台登录的业务场景。当然,目前的这个设计并不完美,仅供参考,感兴趣的小伙伴们可以继续完善这段代码。例如适配器类中的参数目前是设置为String,改为Object[]应该更合理。

学习到这里,相信小伙伴们会有一个疑问:适配器模式与策略模式好像区别不大?我要强调一下,适配器模式主要解决的是功能兼容问题,单场景适配可能不会和策略模式有对比。但复杂场景适配大家就很容易混淆。其实,大家有没有发现一个细节,笔者给每个适配器类都加上了一个support()方法,用来判断是否兼容,support()方法的参数类型也是Object,而support()来自接口。适配器类的实现逻辑并不依赖接口,完全可以将ILoginAdapter接口去掉。而加上接口,只是为了代码规范。上面代码可以说是策略模式、简单工厂模式和适配器模式的综合运用。

相关文章
|
4天前
|
设计模式 存储 安全
Java面试题:设计一个线程安全的单例类并解释其内存占用情况?使用Java多线程工具类实现一个高效的线程池,并解释其背后的原理。结合观察者模式与Java并发框架,设计一个可扩展的事件处理系统
Java面试题:设计一个线程安全的单例类并解释其内存占用情况?使用Java多线程工具类实现一个高效的线程池,并解释其背后的原理。结合观察者模式与Java并发框架,设计一个可扩展的事件处理系统
10 1
|
4天前
|
设计模式 存储 缓存
Java面试题:结合建造者模式与内存优化,设计一个可扩展的高性能对象创建框架?利用多线程工具类与并发框架,实现一个高并发的分布式任务调度系统?设计一个高性能的实时事件通知系统
Java面试题:结合建造者模式与内存优化,设计一个可扩展的高性能对象创建框架?利用多线程工具类与并发框架,实现一个高并发的分布式任务调度系统?设计一个高性能的实时事件通知系统
9 0
|
4天前
|
设计模式 存储 缓存
Java面试题:结合单例模式与Java内存模型,设计一个线程安全的单例类?使用内存屏障与Java并发工具类,实现一个高效的并发缓存系统?结合观察者模式与Java并发框架,设计一个可扩展的事件处理系统
Java面试题:结合单例模式与Java内存模型,设计一个线程安全的单例类?使用内存屏障与Java并发工具类,实现一个高效的并发缓存系统?结合观察者模式与Java并发框架,设计一个可扩展的事件处理系统
9 0
|
2月前
|
测试技术
【一刷《剑指Offer》】面试题 9:斐波那契数列(扩展:青蛙跳台阶、矩阵覆盖)
【一刷《剑指Offer》】面试题 9:斐波那契数列(扩展:青蛙跳台阶、矩阵覆盖)
|
2月前
|
存储 JavaScript
面试题:扩展运算符(...)的优缺点(vue)
面试题:扩展运算符(...)的优缺点(vue)
51 0
|
2月前
|
JavaScript 算法 前端开发
面试题:vue2和vue3区别、vue3项目的打包体积为什么减少40%、vue2和vue3同样可以使用TS开发,为什么vue3就易于扩展呢?vue3的摇树优化是怎么样的优化过程?
面试题:vue2和vue3区别、vue3项目的打包体积为什么减少40%、vue2和vue3同样可以使用TS开发,为什么vue3就易于扩展呢?vue3的摇树优化是怎么样的优化过程?
128 0
|
2月前
|
存储 Linux 调度
[操作系统]秋招面试问到进程扩展知识!!!面试官喜欢的答案
[操作系统]秋招面试问到进程扩展知识!!!面试官喜欢的答案
|
9月前
|
前端开发 Java 数据库连接
【面试题精讲】JVM-类加载器-扩展类加载器
【面试题精讲】JVM-类加载器-扩展类加载器
java202302java学习笔记第二天-数据类型扩展和面试题1
java202302java学习笔记第二天-数据类型扩展和面试题1
31 0
java202302java学习笔记第二天-数据类型扩展和面试题1