本人某个android项目开发阶段一直运行良好,直到上线前夕,在某款跑着android 4.03系统的手机运行却报出一下异常,导致force close:java.lang.IllegalStateException: Can not perform this action after onSaveInstance!
首先得了解一下我那项目的一些基本情况,UI结构是TabActivity包含着5个Tabs,每个tab又是一个独立的Activity。
异常是发生在android 4.03系统上,当我在某个Tab上按Back键时,就会报出java.lang.IllegalStateException: Can not perform this action after onSaveInstanceState
从logout里发现了整个异常发生的过程:
- java.lang.IllegalStateException: Can not perform this action after onSaveInstanceState
- at android.app.FragmentManagerImpl.checkStateLoss(FragmentManager.java:1109)
- at android.app.FragmentManagerImpl.popBackStackImmediate(FragmentManager.java:399)
- at android.app.Activity.onBackPressed(Activity.java:2066)
- at android.app.Activity.onKeyUp(Activity.java:2044)
- at android.view.KeyEvent.dispatch(KeyEvent.java:2529)
- at android.app.Activity.dispatchKeyEvent(Activity.java:2274)
- at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchKeyEvent(PhoneWindow.java:1803)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchKeyEvent(PhoneWindow.java:1855)
- at com.android.internal.policy.impl.PhoneWindow.superDispatchKeyEvent(PhoneWindow.java:1277)
- at android.app.Activity.dispatchKeyEvent(Activity.java:2269)
- at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchKeyEvent(PhoneWindow.java:1803)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.widget.TabHost.dispatchKeyEvent(TabHost.java:297)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at android.view.ViewGroup.dispatchKeyEvent(ViewGroup.java:1112)
- at com.android.internal.policy.impl.PhoneWindow$DecorView.superDispatchKeyEvent(PhoneWindow.java:1855)
- at com.android.internal.policy.impl.PhoneWindow.superDispatchKeyEvent(PhoneWindow.java:1277)
- at android.app.Activity.dispatchKeyEvent(Activity.java:2269)
- at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchKeyEvent(PhoneWindow.java:1803)
- at android.view.ViewRoot.deliverKeyEventPostIme(ViewRoot.java:2880)
- at android.view.ViewRoot.handleFinishedEvent(ViewRoot.java:2853)
- at android.view.ViewRoot.handleMessage(ViewRoot.java:2028)
- at android.os.Handler.dispatchMessage(Handler.java:99)
- at android.os.Looper.loop(Looper.java:132)
- at android.app.ActivityThread.main(ActivityThread.java:4028)
- at java.lang.reflect.Method.invokeNative(Native Method)
- at java.lang.reflect.Method.invoke(Method.java:491)
- at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:844)
- at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:602)
- at dalvik.system.NativeStart.main(Native Method)
上面的异常信息表示,我写的类不是异常的源头。根据异常信息Can not perform this action after onSaveInstanceState,可以了解到异常原因:在onSaveInstanceState行为之后,app执行某个不能响应的行为而导致异常发生。
在信息at android.app.Activity.onBackPressed(Activity.java:2066),这一句表明异常是在响应返回键响应事件的行为上发生的。我们顺藤摸瓜,考究一下在我们按下返回键时,activity会执行的响应:onKeyDown-->onBackPressed-->onPause->onStop->onDestroy。
那导火索onSaveInstanceState又是在什么时候执行的?
我们先看android API的一段原文:
- 先看Application Fundamentals上的一段话:
- Android calls onSaveInstanceState() before the activity becomes vulnerable to being destroyed by the system, but does not bother calling it when the instance is actually being destroyed by a user action
- (such as pressing the BACK key)
从上面可以知道,当某个activity变得“容易”被系统销毁时,该activity的onSaveInstanceState就会被执行,除非该activity是被用户主动销毁的,例如当用户按BACK键的时候。
注意上面的双引号,何为“容易”?言下之意就是该activity还没有被销毁,而仅仅是一种可能性。
onSaveInstanceState的调用遵循一个重要原则,即当系统“未经你许可”时销毁了你的activity,则onSaveInstanceState会被系统调用,这是系统的责任,因为它必须要提供一个机会让你保存你的数据。
那为什么项目里头响应onBackPressed事件时会报出上面的异常呢,还表明是after onSaveInstanceState?
原因是我Tab里面的Activity响应了onBackPressed事件,得弹出task,作为它的父容器TabActivity当然也得弹出task,TabActivity 变得“容易”被系统销毁,于是就调用onSaveInstanceState保存状态。
现在整个流程都明白了,可是,这一切都很正常啊,这个流程也很符合Activity的生命周期啊,为什么还会报异常呢?还是在最新的android 4.03上出问题,难道是说,系统不兼容?
对!
经过一番网上查阅,发现API 11 以上某些控件,包括 Fragment还有ActivityGroup,在调用saveInstanceState
存在Bug,可能是google对
saveInstanceState的实现做过修改。
直到隐藏在后面的原因,解决问题的思路就出来了:让父容器TabActivity在不调用saveInstanceState的情况下onDestroy
具体思路在tab上面的activity监听BACK键的事件,响应并拦截,再通过广播方式通知父容器TabActivity,主动销毁自己,达到原来响应onBackPressed退出App的效果。
相关推荐
这个存储库提供了一种在处理片段传输和后台任务时避免“java.lang.IllegalStateException:Can not perform this action after onSaveInstanceState”的方法。 您可以在的非常权威的阅读有关该问题和可能的解决方案...
### J-LINK V8固件烧录指南 #### 一、引言 J-LINK是一款广泛应用于嵌入式系统开发的编程器与调试器,它能够帮助开发者完成代码的调试和固件的烧录工作。然而,在长时间使用过程中,由于各种原因可能会导致J-LINK...
Library: BibTeX on Android License: Library is free software published under the GNU GENERAL PUBLIC ...You can then perform a debug build by running $ ant debug You can now copy the resulting
The Code Editor provides programmers with an environment in which they can write their code, and perform some editing to it. The various features and functionalities provided by the Code Editor are ...
.setCapabilities(AccessibilityServiceInfoCompat.CAPABILITY_CAN_PERFORM_GESTURES) .setPackageNames(packageName) .build() setServiceInfo(serviceInfo) ``` 以上就是如何在Android Studio 2022.3.1版本中...
在尝试使用INTouch 2014 R2 SP1版本软件在Windows 10操作系统上创建新型应用程序时,用户遇到了“Could not perform operation -- unexpected exception. Unexpected failure (8000ffff)”这一错误提示。该问题出现...
在Android应用开发中,打电话功能是一项常见的需求,尤其是在企业级应用或者个人通讯工具中。本文将深入探讨如何在Android平台上实现打电话功能,并结合Java编程语言进行详细讲解。 首先,我们需要了解的是Android...
<action android:name="android.intent.action.SEARCH" /> android:name="android.app.searchable" android:resource="@xml/searchable" /> ``` 在`res/xml/searchable.xml`中定义搜索属性: ```xml ...
Java种菜源码-Sukrutha-PerformTask是一个基于Java编程语言的开源项目,旨在为开发者提供一个实现任务执行的框架。Sukrutha-PerformTask项目的核心目标是帮助用户高效、可靠地管理一系列任务,使得在复杂的软件工程...
在Android应用开发中,Action Bar是用户界面的一个关键组件,它位于屏幕顶部,提供应用程序的标识、主要操作以及导航选项。然而,系统默认的Action Bar可能无法满足所有设计需求,因此开发者经常需要对其进行自定义...
<action android:name="android.view.InputMethod" /> android:name="android.inputmethodservice" android:resource="@xml/input_method_metadata" /> ``` 至此,我们已经完成了自定义软键盘的创建,包括...
通过调用 AccessibilityNodeInfo 的 `performAction(int action)` 方法,我们可以模拟用户对控件的操作,例如 ACTION_CLICK 或 ACTION_SCROLL_DOWN。 在实际开发中,需要注意的是,由于无障碍服务涉及到用户的隐私...
**J-Link V9 固件修复指南** J-Link 是 SEGGER 公司推出的一款广受欢迎的仿真器和编程器,它支持多种微控制器和嵌入式系统,为开发者提供方便的调试环境。J-Link V9 是该系列的一个版本,可能会遇到需要修复固件的...
1. **Caption**:Action的显示文本。 2. **Category**:分类,用于组织Action,方便管理和查找。 3. **Checked**:选中状态,适用于复选框型Action。 4. **Enabled**:表示Action当前是否可用。 5. **HelpContext**...
1. **Chapter 3: Inside the DEX File** - This chapter delves into the structure and contents of DEX files, providing insights into how decompilers can extract meaningful information from them. ...
### 计算器Android教学实训知识点 #### 一、项目概述 本实训文档旨在通过一个简单的计算器应用开发,帮助学习者掌握Android应用的基本开发流程和技术要点。计算器应用不仅能够加深对Android开发的理解,还能锻炼...
主要介绍了微信小程序控制台提示warning:Now you can provide attr "wx:key" for a "wx:for" to improve performance解决方法,简单分析了wx:for警告提示相关解决方法,需要的朋友可以参考下
DBIx :: Informix :: Perform是Informix Perform查询和更新实用程序的基于Perl和Curses的跨数据库模拟器。 它可以使用现有的执行描述文件,也可以查询数据库的表模式。