现象

  今天工作中,在对公司产品进行测试的时候,程序员小哥点出了一个问题。问题点出的步骤是这样的:

  1.安装APP

  2.点击打开

  3.经过闪屏页,进入主页后,点击HOME键

  4.再次进入程序会重新进入闪屏页,不过经过闪屏页后会停留在HOME前所在的页面

原理分析

  从此我们可以知道QQ安装器其实也就是使用Intent来启动其刚刚安装的那个App,但是问题所在的是:他们的启动Intent并没有跟桌面的启动Intent完全一致!

我们将桌面的Task记为【TaskL】,QQ安装器的Task记为【TaskQ】,我们应用的Task记为【TaskA】,那么分析如下:

  进入桌面: L1 ---- L1是单纯的桌面

  打开QQ: L1Q1Q2 ---- Q2是安装完毕后询问是否启动对应程序的Activity

  点击打开: L1Q1Q2A1A2 ---- A1是入口闪屏页,A2是主页Activity

  返回桌面: Q1Q2A1A2L1 ---- 回到桌面页,也就是L1前置

  点击A的图标: Q1Q2L1A1A2A1 ---- 找到【TaskA】,挪到前台,由于比对Intent并不是完全一致,所以该请求是新启动Activity,那么把A1添加到对应的【TaskA】中

  所以bug出现了,出现了再一次的闪屏页【A1】,问题定位成功!

  PS:这里我稍微变种一下,因为一般我们闪屏页都是在启动主页后finish的,而主页一般是singleTask模式

  打开QQ: L1Q1Q2 ---- Q2是安装完毕后询问是否启动对应程序的Activity

  点击打开: L1Q1Q2A2 ---- A1是入口闪屏页,A2是主页Activity,启动后A1业务逻辑应该finish掉,所以从【TaskA】中挪去

  返回桌面: Q1Q2A2L1 ---- 回到桌面页,也就是L1前置

  点击A的图标: Q1Q2L1A2A1 -> Q1Q2L1A2A1 ---- 找到【TaskA】,挪到前台,由于比对Intent并不是完全一致,所以该请求是新启动Activity,那么把A1添加到对应的【TaskA】中,然后A1所再一次触发启动主页,但是主页是 singleTask模式,所以又回到了上次对应的A2主页,所以现象为再一次出现闪屏页,然后回到原先的主页界面。

解决思路

  自身业务代码规避,我们可以知道,如果是多余的闪屏页入口Activity的话,其基本不可能位于Task的根部,而如果正常启动的话,闪屏页入口Activity必定在多对应的Task的根部位置,那么我们可以从这个地方对于这个bug进行规避,方法就是在闪屏页入口Activity的onCreate代码加入如下一段代码:

// 避免从桌面启动程序后,会重新实例化入口类的activity
if (!this.isTaskRoot()) {
Intent intent = getIntent();
if (intent != null) {
String action = intent.getAction();
if (intent.hasCategory(Intent.CATEGORY_LAUNCHER) && Intent.ACTION_MAIN.equals(action)) {
finish();
return;
}
}
}

  

05-03 22:46