我正在尝试掌握类View
中的onSaveInstanceState
方法(而不是类Activity
中的方法(。该方法确实返回Parcelable
。我从ViewGroup
中得出了自己的View
,并覆盖了该方法以拯救自己的状态。但是当状态要保存时,我得到了一个例外:
java.lang.IllegalStateException:
Derived class did not call super.onSaveInstanceState()
这是真的,但对我来说,仅仅调用该方法似乎还不够。那么我应该怎么做呢?如果该方法传递了一个要写入的Parcel
,我可以简单地将相同的包裹传递给超类,这样事情就会按顺序写入。但这是一个返回值。
是否应该将此返回的对象作为我自己的Parcelable
表示形式的成员包含在内,并在需要时使用 Parcel.writeParcelable
将其与我自己的数据一起封送?或者有没有更好的方法来处理可包裹对象的父调用和链接?如果是这样,在装入超类的实例状态时应该使用什么类装入器?
由于 zapl 没有将他的评论变成答案,所以我这样做了。
有没有更好的方法来处理可包裹对象的父调用和链接?
实现此目的的规范方法是拥有自己的类来保存从View.BaseSavedState
派生的数据,而又从AbsSavedState
派生。可以调用父类的onSaveInstance
处理程序,并将结果对象传递给自己类的构造函数。还原数据时,getSuperState
给出针对父类的实例。
一个典型的代码示例可能如下所示:
static class SavedState extends View.BaseSavedState {
// include own data members here
public SavedState(Parcelable superState) {
super(superState);
}
private SavedState(Parcel in) {
super(in);
// read own data here
}
@Override public void writeToParcel(Parcel out, int flags) {
super.writeToParcel(out, flags);
// write own data here
}
public static final Parcelable.Creator<SavedState> CREATOR =
new Parcelable.Creator<SavedState>() {
public SavedState createFromParcel(Parcel in) { return SavedState(in); }
public SavedState[] newArray(int size) { return new SavedState[size]; }
};
}
@Override public Parcelable onSaveInstanceState() {
SavedState state = new SavedState(super.onSaveInstanceState());
// set data members here
return state;
}
@Override public void onRestoreInstanceState(Parcelable parcelable) {
SavedState state = (SavedState)parcelable;
super.onRestoreInstanceState(state.getSuperState());
// restore from data members here
}
以上内容改编自Cyril Mottier的演讲,但也应该与设计师打算如何使用此类非常匹配。
我是否应该将此返回的对象作为我自己的
Parcelable
表示形式的成员包含在内,并在需要时使用Parcel.writeParcelable
将其与我自己的数据一起封送?
尽管上述似乎是首选,但在幕后,它也依赖于writeParcelable
。因此,如果有理由不使用该基类,只需调用writeParcelable
来存储超类的状态应该没问题。
装入超类的实例状态时应该使用什么类装入器?
AbsSavedState
的当前实现确实使用 null
作为类装入器参数,从而导致使用缺省类装入器。但是,该行代码标有FIXME
注释,因此有一天可能会更改。