目标方法前置检验模型设计与实现

前言

继上次的指纹验证后,产品又提出了个项目典型需求:用户在未登录状态下,点击关注,跳转登录,登录成功后自动执行关注。

分析

这里有两个需求:

  1. 自动跳转到登录界面
  2. 登录成功后再自动执行关注行为

思考下抽象出通用性,首先,我们的目的是执行关注行为,但是关注行为需要用户处于登录状态。也就是说执行某个操作时需要满足一些前提条件,而这些前提条件是需要用户参与才能满足。可能存在多个前提条件的需求,这里可以分离出目标行为和前提条件行为。

技术选型

那就登录页面加个回调喽,登录成功就回调回去,不过,我们可不想在登录界面侵入太多逻辑代码,而且也违背了可维护性和通用性,难道以后扩展时,每个前置条件都要加回调?

再思考,第一想法是利用事件通知机制(EventBus、BroadcastReceiver),把登录成功这个事件 post 出去,异步执行关注行为。但如果前提条件有多个呢,是不是得发送多个事件?是不是得监听多个事件?而且,如果下次需求改了,目标事件不是执行关注事件了,而是执行发帖或者跳转到某个页面,那之前关注行为已经注册了,还是会通知到关注事件,难道要通过 msg 标志位来区分不同的目标行为吗?有没有更优雅简洁的方式呢?

再思考,如果用拦截器呢?

这种行为模式和拦截器非常相似,都是在执行目标前,判断是否符合条件。

但是拦截器似乎并不能直接完成我们的需求,因为我们需要插入一个验证行为后(例如进入登录界面),还要执行相应的操作,保证这个验证行为通过后,才能真正执行我们的目标行为。拦截器执行完后,马上会执行目标方法。中间并不会等待。所以我们根本没有办法去执行我们的登录操作,所以pass了。

那设计模式里的责任链模式,能不能提供点什么思路呢?

在责任链模式里,很多的处理对象由每一个对象对其下家的引用而联接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理此请求。注意,是某一个!而我们的场景是,需要通过所有的前置条件验证,才能执行目标方法。

设计与实现

思考可以改良一下责任链,每一个前置条件对象(Valid类)不再持有其下家的引用,而是采用一个队列存储所有的前置条件对象,以先进先出的顺序依次验证条件。

 private Queue<Valid> validQueue = new ArrayDeque<>();

前置条件对象抽象为接口Valid

public interface Valid {
    /**
     * 是否满足检验器的要求,如果不满足的话,则执行doValid()方法。如果满足,则执行目标action.call
     * @return
     */
    boolean preCheck();

    /**
     * 去执行验证前置行为,例如跳转到登录界面。(但并未完成验证。所以需要在登陆成功时调用preCheck()再次检查)
     */
    void doValid();

}

可以看到,我们把前置条件对象的验证条件和操作分离开来。

目标对象Action就很简单了,仅拥有一个目标执行方法。

public interface Action {
    void call();
}

整体的验证流程,如下图所示:


循环从前置条件对象队列 validQueue 里取出验证对象,调用 check 方法判断是否满足条件,不满足则进入该验证对象的验证流程,验证完毕,再次调用 check 方法判断满足条件,如果满足则取下一个,直至队列为空,就可以直接执行目标方法了。

那么前置条件对象队列 validQueue 和目标对象的持有方,我们封装为 Call,意为一次执行。

public class Call {
    //目标对象 
    private Action action;
    //先进先出验证模型
    private Queue<Valid> validQueue = new ArrayDeque<>();
    //上一个执行的valid
    private Valid lastValid;

}

包装一个单例的持有Call对象的外观类 SingleCall,提供给业务层调用。


public class SingleCall {

    private Call call = new Call();

    public SingleCall addAction(Action action) {
        clear();
        call.setAction(action);
        return this;
    }

    public SingleCall addValid(Valid valid) {
        //只添加无效的,验证不通过的
        if (valid.preCheck()) {
            return this;
        }
        call.addValid(valid);
        return this;
    }

    public void doCall() {
        //如果上一条valid没有通过,是不允许再发起call的
        if (call.getLastValid() != null && !call.getLastValid().preCheck()) {
            return;
        }

        //如果全部都验证通过了,执行action
        if (call.getValidQueue().size() == 0) {
            if (call.getAction() != null) { //容错处理
                call.getAction().call();
                clear();
            }
        } else {
            //执行验证
            Valid valid = call.getValidQueue().poll();
            call.setLastValid(valid);
            valid.doValid();
        }

    }
    .....
}

调用示例(常用场景,单 Action):

Activity 实现 Action 接口,或 new 一个Action 实现类,实现 call 目标行为。

SingleCall.getInstance()
          .addAction(ActionActivity.this)
          .addValid(new LoginValid())//前置条件,可能有多个
          .addValid(new OtherValid()
          .doCall();

前置行为完成后,调用 SingleCall.getInstance().doCall(); 重新启动验证模型。因为验证操作需要用户手动触发完成,我们只是引导用户到了验证体里,由于我们因为等待用户的操作,验证模型就在这里停下来了,如果验证操作成功了,我们需要让整个验证模型再运转起来了,所以验证后,永远少不了手动开启验证模型。

以下是整个模型的类 UML 图,重点梳理 Valid、Call、Action 三者的关系:


类关系图

应用场景

源码地址

github 源码地址

  • 增加了容错处理
  • 补充了嵌套 Call 的情况

参考:


我是 FeelsChaotic,一个写得了代码 p 得了图,剪得了视频画得了画的程序媛,致力于追求代码优雅、架构设计和 T 型成长。

欢迎关注 FeelsChaotic 的简书掘金,如果我的文章对你哪怕有一点点帮助,欢迎 ❤️!你的鼓励是我写作的最大动力!

最最重要的,请给出你的建议或意见,有错误请多多指正!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,657评论 6 505
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,889评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,057评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,509评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,562评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,443评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,251评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,129评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,561评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,779评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,902评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,621评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,220评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,838评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,971评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,025评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,843评论 2 354

推荐阅读更多精彩内容

  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,967评论 6 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,656评论 18 139
  • #幸福是需要修出来的~每天进步1%~幸福实修09班~01~蔷薇 20170816(29/30)09班 【幸福两朵玫...
    幸福实修蔷薇阅读 188评论 0 0
  • 2014-12-02 想了很久以什么方式记录这次旅行,可惜各种花俏最后还是被以下的流水帐打败,很多不同的经历和感受...
    Nicole雅斐阅读 505评论 0 0
  • 在遇见你之前,我的人生可以说是黑暗的。不懂拒绝、没有主见、没有自己,这是那时的我。 遇见你以后,我开始学会做自己,...
    小万管家阅读 1,765评论 2 0