54

给定一个自定义类org.example.app.MyClass implements Parcelable,我想写List<MyClass>一个包裹。我做了编组

 List<MyClass> myclassList = ...
 parcel.writeList(myclassList);

每当我尝试用

 List<MyClass> myclassList = new ArrayList<MyClass>();
 parcel.readList(myclassList, null);

有一个"BadParcelableException: ClassNotFoundException when unmarshalling org.example.app.MyClass"例外。

这里有什么问题?为什么找不到类?

4

4 回答 4

110

不要使用当您null作为 ClassLoader 参数提供时使用的框架类加载器解组自定义类(即由您的应用程序提供而不是由 Android 框架提供的类)。使用应用程序类加载器:

parcel.readList(myclassList, getClass().getClassLoader());

每当一个Parcel.read*()方法也有一个 ClassLoader 作为参数(例如Parcel.readList(List outVal, ClassLoader loader))并且您想从 中读取应用程序类时Parcel,请使用可以使用 检索的应用程序类加载器getClass().getClassLoader()

背景: Android 带有两个类加载器:系统类加载器,它能够加载除应用程序提供的系统类之外的所有系统类;和应用程序类加载器,它已将系统类加载器设置为其父类,因此能够加载所有类。如果你给 null 作为类加载器,将使用框架类加载器,导致ClassNotFoundExceptionParcel.readParcelableCreator()

感谢 alexanderblom 提供了引导我走上正轨的提示。

于 2013-08-08T12:38:42.173 回答
19

我相信更正确的形式是:

List<MyClass> myclassList = new ArrayList<MyClass>();
parcel.readList(myclassList, MyClass.class.getClassLoader());

因为在这里您明确地使用了 List 的泛型类型的类加载器。

于 2014-01-29T07:22:02.063 回答
7

我遇到了同样的问题,而不是 parcleable 我使用了 bundle

intent.setExtrasClassLoader(getClassLoader());
/* Send optional extras */
Bundle bundle = new Bundle();
bundle.putParcelable("object", mObject);
bundle.putString("stringVal", stringVal);

intent.putExtra("bundle",bundle);

在我的意图类中,我使用它来检索值:

Bundle oldBundle = intent.getBundleExtra("bundle");
ResultReceiver receiver = oldBundle.getParcelable("object");
String stringVal = oldBundle.getString("stringVal");
intent.setExtrasClassLoader(getClassLoader());

希望对某些人有所帮助。

于 2015-09-12T13:31:14.310 回答
5

如果您错误地将自定义视图子类化,您可能会收到此错误。

假设您是子类BottomNavigationView化,并且您想将保存的状态添加到onSaveInstanceState().

Parcelable 样板的错误实现(从另一个类或模板复制)如下所示:

static class State extends BaseSavedState {

    Bundle stateBundle;

    //incorrect as super state uses ClassLoaderCreator
    public static final Creator<State> CREATOR = new Creator<State>() {
        public State createFromParcel(Parcel in) {
            return new State(in);
        }

        public State[] newArray(int size) {
            return new State[size];
        }
    };

    State(Parcel source) {
        super(source);
        this.stateBundle = source.readBundle(getClass().getClassLoader());
    }

    State(Parcelable superState) {
        super(superState);
    }

    @Override
    public void writeToParcel(Parcel out, int flags) {
        super.writeToParcel(out, flags);
        out.writeBundle(stateBundle);
    }
}

这不起作用,因为超状态BottomNavigationView需要类加载器。相反,您应该仔细检查SavedStateBottomNavigationView并使用正确的ClassLoaderCreator而不是Creator

static class State extends AbsSavedState {

    Bundle stateBundle;

    public static final Creator<State> CREATOR = new ClassLoaderCreator<State>() {
        public State createFromParcel(Parcel in, ClassLoader classLoader) {
            return new State(in, classLoader);
        }

        @Override
        public State createFromParcel(Parcel source) {
            return new State(source, null);
        }

        public State[] newArray(int size) {
            return new State[size];
        }
    };

    State(Parcel source, ClassLoader classLoader) {
        super(source, classLoader);
        this.stateBundle = source.readBundle(classLoader);
    }

    State(Parcelable superState) {
        super(superState);
    }

    @Override
    public void writeToParcel(Parcel out, int flags) {
        super.writeToParcel(out, flags);
        out.writeBundle(stateBundle);
    }
}

请注意,扩展android.support.v4.view.AbsSavedState可能是比BaseSavedState或更好的选择,android.view.AbsSavedState因为它允许您将类加载器传递给超类:

SavedState(Parcel source, ClassLoader classLoader) {
    super(source, classLoader); //available in android.support.v4.view.AbsSavedState
    this.stateBundle = source.readBundle(classLoader);
}
于 2018-01-23T23:19:43.823 回答