Android片段和方向更改导致:IllegalStateException:无法在onSaveInstanceState之后执行此操作

时间:2012-07-27 09:17:46

标签: android fragment illegalstateexception

每当我的主要活动加载了片段并且用户启动新活动,切换设备的方向并返回主活动时,我就会收到此错误。

@Override
public void onCreate(Bundle savedInstanceState) {
    setContentView(R.layout.home_layout);
    super.onCreate(savedInstanceState);
    fragmentManager = getSupportFragmentManager();
    fragment = fragmentManager.findFragmentById(R.id.layFragment);

    initialize();
}

@Override
public void onConfigurationChanged(Configuration newConfig) {
    setContentView(R.layout.home_layout);
    initialize();
    super.onConfigurationChanged(newConfig);
}

private void initialize() {
    layStatus = (LinearLayout) findViewById(R.id.layStatus);
    txtStatus = (TextView) findViewById(R.id.txtStatus);
    ....
    handleFragments(lastFragmentId);
}

public void handleFragments(int fragmentId) {
        if (fragment == null) {
            FragmentTransaction ft = fragmentManager.beginTransaction();
            if (fragmentId==someFragmentId){
                ft.replace(R.id.layFragment, new FragmentSomeFragment());
            }
            else
            ....

            ft.commit();
        }
}

在我的Android清单中,活动被声明为:

 <activity
        android:name=".HomeActivity"
        android:configChanges="keyboardHidden|orientation" />
<activity

在另一个关于SO的问题中,我发现这可能是由支持库中的一个错误造成的,我是否添加了运气:

// needed as a workaround for a bug in the Support library
@Override
protected void onSaveInstanceState(Bundle outState) {
    outState.putString("WORKAROUND_FOR_BUG_19917_KEY", "WORKAROUND_FOR_BUG_19917_VALUE");
    super.onSaveInstanceState(outState);
}

我的应用程序从Android 2.2运行,我正在使用android-support-v4.jar支持库来获取片段。

日志如下:

07-27 11:56:20.399: E/AndroidRuntime(16021): FATAL EXCEPTION: main
07-27 11:56:20.399: E/AndroidRuntime(16021): java.lang.IllegalStateException: Can not perform this action after onSaveInstanceState
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.support.v4.app.FragmentManagerImpl.checkStateLoss(FragmentManager.java:1299)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.support.v4.app.FragmentManagerImpl.enqueueAction(FragmentManager.java:1310)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.support.v4.app.BackStackRecord.commitInternal(BackStackRecord.java:541)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.support.v4.app.BackStackRecord.commit(BackStackRecord.java:525)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at com.rightcab.driver.core.HomeActivity.handleFragments(HomeActivity.java:341)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at com.rightcab.driver.core.HomeActivity.initialize(HomeActivity.java:128)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at com.rightcab.driver.core.HomeActivity.onConfigurationChanged(HomeActivity.java:153)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.app.ActivityThread.performConfigurationChanged(ActivityThread.java:3618)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.app.ActivityThread.handleActivityConfigurationChanged(ActivityThread.java:3771)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1328)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.os.Handler.dispatchMessage(Handler.java:99)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.os.Looper.loop(Looper.java:137)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at android.app.ActivityThread.main(ActivityThread.java:4745)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at java.lang.reflect.Method.invokeNative(Native Method)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at java.lang.reflect.Method.invoke(Method.java:511)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:789)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:556)
07-27 11:56:20.399: E/AndroidRuntime(16021):    at dalvik.system.NativeStart.main(Native Method)

4 个答案:

答案 0 :(得分:16)

首先,正如我所看到的,您希望自己处理配置更改。为了让API Level 13+能够正常使用,您必须再向configChanges参数添加一个值,如here所述。

接下来,当用户离开您的主要活动时,会为其调用onSaveInstanceStateonPause方法。当用户旋转设备并返回主要活动时。 onConfigurationChanged方法在 onResume()之前称为。因此,您的活动仍然暂停,您无法执行FragmentTransaction

此外,如果我们查看源代码,我们可以看到onResume方法的以下注释:

  

将onResume()发送到片段。请注意,为了更好   与平台的旧版本互操作   此调用附加到活动的片段是   恢复。这意味着在某些情况下,之前的状态可能仍然存在   保存,不允许修改状态的片段事务。   要正确地与处于正确状态的片段进行交互,您应该这样做   而是覆盖{@link #onResumeFragments()}。

因此,在您的活动中操作片段的正确位置是覆盖onResumeFragments方法,因为我们可以在源代码中阅读此方法的注释:

  

这是{@link #onResume()}的片段导向版本   您可以覆盖以在同一个Activity中执行操作   它的片段恢复的点。一定要经常打电话   超级班。

protected void onResumeFragments() {
    super.onResumeFragments();

    // YOUR STUFF IS HERE
}

答案 1 :(得分:6)

如果StenaviN建议在返回您的活动时在resume()之前返回onConfigurationChange()

这是生命周期:

onCreate()
onResume()
// Move away from you're Activity
onPause()
// Move back to your Activity
onConfigurationChange()
onResume()

但重要的是这个:

如果您恢复Activity或更改Orientation的{​​{1}},则Activity会很好!你不需要用新副本替换旧的副本,事实上你不应该!如果您只是删除此行,则不会出现问题:

Fragments

然而如果你这样做是因为你需要handleFragments(lastFragmentId); 加载新的布局资源(Fragment =&gt; layout/frag.xml)那么你'需要做这样的事情:

layout-land/frag.xml

答案 2 :(得分:1)

您使用的是最新版本的support-v4库吗?它解决了我类似的问题。

答案 3 :(得分:1)

如果你坚持使用支持库的r7版本(例如因为你正在使用maven并且desesperatly等待更新...;)),那么你可以使用onPostResume来避免这个问题。如果您的版本是r11或更高版本,则可以切换到onResumeFragements