现象
今天工作中,在对公司产品进行测试的时候,程序员小哥点出了一个问题。问题点出的步骤是这样的:
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; } } }