今天我要和大家分享的是安卓的一个问题,极少数人知道,我也是在之前做完项目的时候,测试提出来的,这也得非常感谢这么细心的测试妹妹,但是当时这个问题提给我后,我的内心是拒绝的,并以这是系统原生的问题扔回去了,但是这并不是程序员的作风嘛,有问题就得解决问题,所以后来我还是偷摸的上网查了一些相关资料,终于是不负有心人啊,碰到一样的情况了,后来在一些大牛的指导下,仅仅是加了一行代码,整个问题就解决了,下面我来描述这个问题的现象:
从安卓各大应用市场下载并安装安卓应用,安装完成后,当前界面下方会出现“打开”按钮,这时候我们点击“打开”,会启动应用,进入到应用的起始页,此时我们什么也不干,按home键返回到桌面,找到应用图标所在区域,点击应用图标,此时我们所期待的现象是重新回到之前我们打开的页面,对吗?然后这个时候你点击桌面上的应用图标打开应用,这个时候你会惊奇的发现,什么?应用重新启动了?按返回键你退出应用一次,然后又回到了之前启动的应用页面了?没错,这就是问题所在!我的任务就是教大家如何解决这个问题。
通过上面现象描述,大家应该看懂了问题的现象了吧,我们通过一个具体的实现来重现一下这个问题,首先我们打开“应用宝”(这里绝不是做广告,你们也可以打开其他的安卓应用市场),搜索“收益计算宝”(当然其他的应用也可以啦,基本上应用没注意到这个细节都存在这个问题),下载后如图:
这时候出现了“打开”按钮,我们点击打开按钮启动应用进入主界面,此时我们按照之前的操作,按home键返回桌面,然后在桌面上找到应用图标点击进入就会发现会重启应用,并且退出的时候退出两次才会回到桌面,那么我们来分析一下这是为什么呢?当然,收益计算宝这款软件并不会,因为我做过处理了,所以各位想试验的,可以随便挑一款软件来测试。
首先我们要知道,一个task并不是对应一个线程,实际上一个task可以包含多个进程, 一个Task中可能有多个同一个Activity的实例,然后这时候第一次的情况下,并不是设置launchMode就可以保证单例的,在资料上看到,这也是安卓系统的一个漏洞,但是官方也给出了对应的API来解决这个问题,但是实际上这个问题大家并未给足够多的关注,所以很多人不知道这个现象。
但是这种现象在以root静默安装的方式和用Eclipse运行安装的情况下并不会出现,很多人拿launchMode说事,但是实际上没有任何作用,况且从那四个属性的描述上,可以看出也与此无关。
解决方法:
说了那么多废话,下面给出解决方案,在你的应用的起始页,记住一定要是程序打开的第一个页面,在onCreate方法里,在super.onCreate(savedInstanceState)这一行代码之后,一定要在这行代码之后,插入以下代码:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (!isTaskRoot()) {
finish();
return;
} else {
setContentView(R.layout.activity_main);
}
}
if(!isTaskRoot()),判断该Activity是不是任务空间的源Activity,“非”也就是说是被系统重新实例化出来,如果你就放在launcher Activity中话,这里可以直接return了。
来源google : https://code.google.com/p/android/issues/detail?id=14262
所以到现在,你们觉得今天我分享的内容够给力吗?仅仅一行代码的问题,就可以解决这个恼人的bug,快分享给你的伙伴们吧,如果有问题,欢迎留言。