Fragment Can not perform this action after onSaveInstanceState

更多博客 https://70kg.info

最近遇到了这个异常,记录一下解决办法。

首先看一下这个异常是从哪里抛出来的:

//FragmentManager
 public void enqueueAction(OpGenerator action, boolean allowStateLoss) {
        if (!allowStateLoss) {
            checkStateLoss();
        }
        ....
    }

   private void checkStateLoss() {
        if (mStateSaved) {
            throw new IllegalStateException(
                    "Can not perform this action after onSaveInstanceState");
        }
        if (mNoTransactionsBecause != null) {
            throw new IllegalStateException(
                    "Can not perform this action inside of " + mNoTransactionsBecause);
        }
    }

也就是在对fragment进行commit的时候,如果这时候mStateSaved那就会抛出这个异常,而mStateSaved是在saveAllState也就是onSaveInstanceStatedispatchStop也就是Activity::onStop中。也就是说如果在这两个生命周期方法之中或者之后对Fragment进行Commit就会出现这个异常。特别是DialogFragment,一般的Fragment还可以使用commitAllowingStateLoss来进行”丑陋的”避免,而DialogFragment的默认show方法

    public void show(FragmentManager manager, String tag) {
        mDismissed = false;
        mShownByMe = true;
        FragmentTransaction ft = manager.beginTransaction();
        ft.add(this, tag);
        ft.commit();
    }

根本不给机会。

上面说到抛出这个异常的原因,引申出什么情况下容易出现这种情况,就是异步回调的时候进行commit。因为这个时候你无法知道当前的状态,碰巧这个时候应用进入后台,执行了onStop或者被杀死onSaveInstanceState,就挂了。使用Fragment有好处,就是比Activity轻,跳转不用进行IPC。但是真的不怎么好用,到处的坑。再说这里,例如使用MVP,在onDestroyViewView进行清空,(这个是在ActivityonDestroy调用),也就是在这之前对View操作都是安全的,但是FragmentonStop之后就不能再进行commit了。使用View==null判断根本不行,真的防不胜防������。

解决办法:
Fragment全解析系列(一):那些年踩过的坑但是貌似作者没有考虑onStop的情况。至于为什么前面说commitAllowingStateLoss是个”丑陋”的解决办法,这里有更详细的说明Fragment Transactions & Activity State Loss,早在13年就提及这个问题。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值