无意中发现的一个bug,测试了一下很多设备上很多APP都有这种情况,包括部分知名公司产品,研究了一下发现只是很早就存在的一个系统bug,以前竟然没能留意到。
如何重现
浏览器下载、QQ下载或者通过软件市场下载安装后,直接在安装成功页面点击打开,程序正常启动闪屏页面,然后进入主页面。此时按“Home”键让程序切后台,再在桌面上点击此应用的图标,理论上应该直接恢复到刚才的主页面,结果此时程序重新显示闪屏页,然后进入主页。通过一路按返回,可以发现每次按“Home”恢复时都是重新创建了新的Activity。
这个问题说起来也是比较严重的,用户安装了一个应用,启动然后手机注册,程序切后台查看短信注册码,再切回去发现程序从头走了。如果一路back让程序退出,再次从桌面icon启动应用则不会出现问题
分析
这个问题第一反应就是看下Activity的flag,在闪屏页打一个Log,发现安装后直接打开的Flag值为0x10000000,即FLAG_ACTIVITY_NEW_TASK,此时按Home切后台再按桌面Icon,闪屏页面的Flag值为0x10600000,即这个Flag导致的这个奇怪的问题。而程序退出正常从桌面Icon启动的Flag应该为0x10200000(FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_RESET_TASK_IF_NEEDED)。上面出问题的那个Flag中的值600000再类Intent中并没有找到,但是通过位运算不难发现它是200000和400000的或值,即0x10600000的值为(FLAG_ACTIVITY_NEW_TASK | FLAG_ACTIVITY_RESET_TASK_IF_NEEDED | FLAG_ACTIVITY_BROUGHT_TO_FRONT)。通过搜索发现这是一个系统bug,Issue Tracker上面很早都有人提出这个问题了,但一直没有解决。我找了一下,也没能找到引起这个问题的原因,大家有谁研究过这一块的源码找到原因的希望能给我讲一下。
解决方案
问题的解决方案已经明确了,就是在我们的root activity Flag值为0x10600000的这种情况不处理即可。stackoverflow中和Issue Tracker中都有人提出了一个认可程度比较高的解决方案,即在应用的root acitivity的onCreate()中加入以下代码:
if (!isTaskRoot()) {
Intent intent = getIntent();
String action = intent.getAction();
if (intent.hasCategory(Intent.CATEGORY_LAUNCHER) && action != null && action.equals(Intent.ACTION_MAIN)) {
finish();
return;
}
}
这个是从判断是否root task来处理的,通过打log可以确定问题出现时虽然每次运行都是都是新建一个闪屏页面,但是它的taskId并没有改变,所以这个方案不失为一个不错的解决方案。
在最开始分析这个问题原因的时候,反编译了一些不能重新此问题的APP,例如美团外卖,我在它的闪屏页面里面的onCreate()函数里面发现如下代码:
if ((getIntent().getFlags() & 0x400000) != 0) {
finish();
return;
}
这个方法与我们上面的分析不谋而合,通过简单测试这两种方法的都能解决问题。