我出现这个问题的原因就是,在当前Acitivity 操作的过程中 跳到另外一个界面做了一些操作然后返回并带了一些参数,再在当前Activity 显示DialogFrament 展示带过来的参数,此时崩溃出现了这个错误信息
IllegalStateException: Can not perform this action after onSaveInstanceState
出现这个问题的原因大致就是在当前Activity 操作的过程中,跳转到其它视图做了一些操作,此时当前Activity运行了onSaveInstanceState方法,保存了当前Acitivity 的一些状态。而又在此时从另一个视图返回了,再展示Fragment 时便会出错。也就是再Activity运行完onSaveInstanceState后再添加Fragment 相关视图就会报出异常,具体分析太难了我也不懂,可以看这个https://blog.csdn.net/ranxiedao/article/details/8214936
官方给出的解决方案就是使用DialogFragment的commitAllowingStateLoss的方法显示。翻译过来的意思也就是“允许损失状态”的。但是这个方法被隐藏了。我们调用不了。于是乎就只能采用反射机制了。我找了很多文章,其中有一篇些的非常好给出了示例和分析。但是找不大了。
第一步:就是根据DialogFragment的包路径,添加一个一模一样的包路径。
第二步:在这个里面重写show()方法,并用反射回去两个属性设置上值。FragmentTransaction 调用 commitAllowingStateLoss 来显示对话框。
public class BDialogFragment extends DialogFragment {
View rootView;
Context mContext;
public View getRootView(ViewGroup container, int resId) {
mContext = getActivity();
rootView = LayoutInflater.from(mContext).inflate(resId, container, false);
return rootView;
}
public <T extends View> T obtainView(int resId){
if(null == rootView){
return null;
}
return (T) rootView.findViewById(resId);
}
/**
* 主要时这个方法。上面两个方法只是获取布局用的,可以不要
* @param manager
* @param tag
*/
@Override
public void show(FragmentManager manager, String tag) {
try{
Field dismissed = DialogFragment.class.getDeclaredField("mDismissed");
dismissed.setAccessible(true);
dismissed.set(this,false);
} catch (NoSuchFieldException | IllegalAccessException e) {
e.printStackTrace();
}
try {
Field showByMe = DialogFragment.class.getDeclaredField("mShownByMe");
showByMe.setAccessible(true);
showByMe.set(this,true);
} catch (NoSuchFieldException | IllegalAccessException e) {
e.printStackTrace();
}
FragmentTransaction ft = manager.beginTransaction();
ft.add(this, tag);
ft.commitAllowingStateLoss();
}
}
第三步,用自己原先继承DialgoFragment 的类,继承这个BDialogFragment类就好了。而这里面为啥要修改这两个属性的值可以看这个文章里的分析:https://www.jianshu.com/p/bcc5af0085d3?t=1481296807683