当Activity跳转偶遇单身多年的老汉

本文章已授权鸿洋微信公众号转载:当Activity跳转偶遇单身多年的老汉

问题介绍

在项目中,Activity多重跳转一直是开发中最常见的问题,网上的解决方案很多,但是要怎么解决才是最佳的往往才是头疼的问题,我现在要讲的是如何真正的解决这个问题而不留一丝Bug,先介绍几种已有的方案以及优缺点

AOP 面向切面

这里不讲 AOP 的集成,如需了解请左拐百度,这里只讲优势和劣势

textView.setOnClickListener(new OnClickListener() {

    @EnableFastOnClick
    @Override
    public void onClick(View v) {
        
    }
});
  • 优点:对 View 点击事件的方法进行注解,看起来比较简洁

  • 缺点:每一处 View 点击事件都要进行注解,开发成本较高,容易出现遗漏

Activity 启动模式

<activity
    android:name=".ui.activity.XXXActivity"
    android:launchMode="singleTop" />

为 Activity 文件中设置 singleTop,这里复习一下 singleTop 启动模式

singleTop:单一顶部模式,如果任务栈的栈顶存在这个要开启的 Activity,不会重新的创建 Activity,而是复用已经存在的 Activity。保证栈顶如果存在,不会重复创建

  • 优点:直接在清单文件中设置 Activity 的启动模式,简单粗暴

  • 缺点:每新增 Activity 都要设置启动模式,并且只能指定singleTop,开发成本较高,容易出现遗漏

startActivity 拦截

首先,我们需要先造一个双击判断工具类

public final class DoubleClickHelper {

    private static final long[] TIME_ARRAY = new long[2]; // 数组的长度为2代表只记录双击操作

    /**
     * 是否在短时间内进行了双击操作
     */
    public static boolean isOnDoubleClick() {
        // 默认间隔时长
        return isOnDoubleClick(1500);
    }

    /**
     * 是否在短时间内进行了双击操作
     */
    public static boolean isOnDoubleClick(int time) {
        System.arraycopy(TIME_ARRAY, 1, TIME_ARRAY, 0, TIME_ARRAY.length - 1);
        TIME_ARRAY[TIME_ARRAY.length - 1] = SystemClock.uptimeMillis();
        return TIME_ARRAY[0] >= (SystemClock.uptimeMillis() - time);
    }
}

重写 Activity 的 startActivity 方法

public abstract class BaseActivity extends AppCompatActivity {

    @Override
    public void startActivity(Intent intent) {
        if (DoubleClickHelper.isOnDoubleClick(500)) {
            return;
        }
        super.startActivity(intent);
    }
}

这样写其实存在一个漏洞,让我们看 Activity 的跳转方法

我想大家的第一眼感觉是和我一样的,这是神马?我难道要重写那么多个?

遇到这种问题,一般菜鸟抱大腿的流程:

  • 菜鸟:遇到不会的问题怎么办?

  • 老鸟:不会百度啊!百度不会吗?

  • 菜鸟:百度不行怎么办?

  • 老鸟:百度不行就换谷歌啊!

  • 菜鸟:谷歌也不行怎么办?

  • 老鸟:源码是最好的老师!

这里只是讲个段子,接下来让我们通过查看源码来解决这个问题,先看 startActivity 的源码

这里调用了同名不同参的方法,再看

原来 startActivity 最终还是要回调 startActivityForResult


从这里看到 startActivityForResult 两个方法,参数短的方法还是调用了参数长的方法,这里我们只需要重写那个参数长的方法即可,那我们不能用刚刚那种方式了,把 startActivity 换成 startActivityForResult

public abstract class BaseActivity extends AppCompatActivity {

    @Override
    public void startActivityForResult(Intent intent, int requestCode, @Nullable Bundle options) {
        if (DoubleClickHelper.isOnDoubleClick(500)) {
            return;
        }
        super.startActivityForResult(intent, requestCode, options);
    }
}

其实这样还存在一个问题,如果这个界面需要多重跳转怎么办呢?这样直接写死 BaseActivity 是不是不利于扩展?

这个问题解决也很简单,在 BaseActivity 预留一个方法,子类可以重写这个方法来决定是否要检查和判断 Activity 多重跳转的问题

public abstract class BaseActivity extends AppCompatActivity {

    @Override
    public void startActivityForResult(Intent intent, int requestCode, @Nullable Bundle options) {
        if (isCheckActivityJump() && DoubleClickHelper.isOnDoubleClick(500)) {
            return;
        }

        // 查看源码得知 startActivity 最终也会调用 startActivityForResult
        super.startActivityForResult(intent, requestCode, options);
    }

    /**
     * 是否检查 Activity 跳转频率,避免重复跳转
     */
    protected boolean isCheckActivityJump() {
        // 默认需要检查和判断
        return true;
    }
}

其实就这两句代码,非常简单,接下来总结一下

  • 优点:基类处理,一劳永逸,开发成本极低

  • 缺点:不能精准的判断跳转的 Activity 是否是重复的,也就是说如果同时跳转两个不同的 Activity,结果只有第一个成功跳转,而第二个却没有跳转

startActivityForResult 拦截优化

上一个解决方案还残留着Bug,追求完美的我们怎能容许这种事情的发生,接下来让我们来给这个问题画上圆满的句号

首先要想知道重复跳转的 Activity 是不是同一个,我们可以通过 Intent 这个对象来进行判断,不过在此之前我们要先复习一下 Activity 的启动方式

  • 显式意图启动

    • 构造方法:new Intent(Context packageContext, Class<?> cls)

    • 对象方法:intent.setClass(Context packageContext, Class<?> cls)

  • 隐式意图启动

    • 构造方法:new Intent(String action)

    • 对象方法:intent.setAction(String action)

这里已经列出这两种启动方式的使用了,我们可以利用显式意图和隐式意图来分别创建一个 Tag 标记,用于判断跳转的 Activity 是否是重复的

// 标记对象
String tag;
if (intent.getComponent() != null) { // 显式跳转
    tag = intent.getComponent().getClassName();
}else if (intent.getAction() != null) { // 隐式跳转
    tag = intent.getAction();
}

除了判断是否重复了之外,还需要再判断跳转时间间隔

if (tag.equals(mActivityJumpTag) && mActivityJumpTime >= SystemClock.uptimeMillis() - 500) {
    // 检查不通过
    result = false;
}

完整代码如下

public abstract class BaseActivity extends AppCompatActivity {

    @Override
    public void startActivityForResult(Intent intent, int requestCode, @Nullable Bundle options) {
        if (startActivitySelfCheck(intent)) {
            // 查看源码得知 startActivity 最终也会调用 startActivityForResult
            super.startActivityForResult(intent, requestCode, options);
        }
    }

    private String mActivityJumpTag;
    private long mActivityJumpTime;

    /**
     * 检查当前 Activity 是否重复跳转了,不需要检查则重写此方法并返回 true 即可
     *
     * @param intent          用于跳转的 Intent 对象
     * @return                检查通过返回true, 检查不通过返回false
     */
    protected boolean startActivitySelfCheck(Intent intent) {
        // 默认检查通过
        boolean result = true;
        // 标记对象
        String tag;
        if (intent.getComponent() != null) { // 显式跳转
            tag = intent.getComponent().getClassName();
        }else if (intent.getAction() != null) { // 隐式跳转
            tag = intent.getAction();
        }else {
            return result;
        }

        if (tag.equals(mActivityJumpTag) && mActivityJumpTime >= SystemClock.uptimeMillis() - 500) {
            // 检查不通过
            result = false;
        }

        // 记录启动标记和时间
        mActivityJumpTag = tag;
        mActivityJumpTime = SystemClock.uptimeMillis();
        return result;
    }
}

此解决方案已经加入啃得香豪华套餐:AndroidProject,欢迎各位提 issue,欢迎 star

Android 技术讨论 Q 群:10047167

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

推荐阅读更多精彩内容