Android中Bundle的错误幻数

col*_*ore 12 java android parcelable

我使用以下代码将数据从一个活动传递到其他活动:

 @Override
    public void execute(List<Report> reports, Questions question) {
        Intent replyIntent = new Intent(listener, ReplyActivity.class);
        replyIntent.putExtra("id", 0L);
        replyIntent.putExtra("questions", question);
        listener.openReportOk(question);
        listener.startActivity(replyIntent);
    }
Run Code Online (Sandbox Code Playgroud)

监听器是回调的活动参考.

问题是这堂课:

@Table(name = "Questions")
public class Questions extends Entity implements Parcelable {

    public static final Creator<Questions> CREATOR = new Creator<Questions>() {
        public Questions createFromParcel(Parcel source) {
            return new Questions(source);
        }

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

    @TableField(name = "idReport", datatype = DATATYPE_INTEGER)
    private int idReport;
    @TableField(name = "nameReport", datatype = DATATYPE_STRING)
    private String nameReport;
    @TableField(name = "replyGroups", datatype = DATATYPE_STRING)
    private String replyGroups;
    @TableField(name = "questionGroups", datatype = DATATYPE_ENTITY)
    private List<QuestionsGroup> questionsGroup;
    private Boolean canCreateNew;

    public Questions(int idReport, String nameReport, String replyGroups, List<QuestionsGroup> questionsGroup) {
        this.idReport = idReport;
        this.nameReport = nameReport;
        this.replyGroups = replyGroups;
        this.questionsGroup = questionsGroup;
        this.canCreateNew = false;
    }

    public Questions() {
        questionsGroup = new ArrayList<QuestionsGroup>();
    }

    private Questions(Parcel in) {
        this();
        this.idReport = in.readInt();
        this.nameReport = in.readString();
        this.replyGroups = in.readString();
        Bundle b = in.readBundle(QuestionsGroup.class.getClassLoader());
        this.questionsGroup = b.getParcelableArrayList("questionGroups");
        this.canCreateNew = (Boolean) in.readValue(Boolean.class.getClassLoader());
    }


    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(this.idReport);
        dest.writeString(this.nameReport);
        dest.writeString(this.replyGroups);
        Bundle b = new Bundle();
        b.putParcelableArrayList("questionGroups", (ArrayList<QuestionsGroup>) this.questionsGroup);
        dest.writeBundle(b);
        dest.writeValue(this.canCreateNew);
    }
}
Run Code Online (Sandbox Code Playgroud)

当我收到onCreate方法的包裹时:

 private void getData(Intent data) {
        ID = data.getExtras().getLong("id");
        questions = data.getExtras().getParcelable("questions");
    }
Run Code Online (Sandbox Code Playgroud)

我得到这个错误:

10-05 13:19:15.508    3499-3499/com.firext.android E/AndroidRuntime? FATAL EXCEPTION: main
    Process: com.firext.android, PID: 3499
    java.lang.RuntimeException: Unable to start activity ComponentInfo{com.firext.android/com.firext.android.activities.reply.ReplyActivity}: java.lang.IllegalStateException: Bad magic number for Bundle: 0x28
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2255)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2317)
            at android.app.ActivityThread.access$800(ActivityThread.java:143)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1258)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5070)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:836)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:631)
     Caused by: java.lang.IllegalStateException: Bad magic number for Bundle: 0x28
            at android.os.BaseBundle.readFromParcelInner(BaseBundle.java:1342)
            at android.os.BaseBundle.<init>(BaseBundle.java:90)
            at android.os.Bundle.<init>(Bundle.java:66)
            at android.os.Parcel.readBundle(Parcel.java:1645)
            at com.firext.android.domain.QuestionsGroup.<init>(QuestionsGroup.java:53)
            at com.firext.android.domain.QuestionsGroup$1.createFromParcel(QuestionsGroup.java:25)
            at com.firext.android.domain.QuestionsGroup$1.createFromParcel(QuestionsGroup.java:23)
            at android.os.Parcel.readParcelable(Parcel.java:2160)
            at android.os.Parcel.readValue(Parcel.java:2066)
            at android.os.Parcel.readListInternal(Parcel.java:2422)
            at android.os.Parcel.readArrayList(Parcel.java:1756)
            at android.os.Parcel.readValue(Parcel.java:2087)
            at android.os.Parcel.readArrayMapInternal(Parcel.java:2393)
            at android.os.BaseBundle.unparcel(BaseBundle.java:221)
            at android.os.Bundle.getParcelableArrayList(Bundle.java:782)
            at com.firext.android.domain.Questions.<init>(Questions.java:58)
            at com.firext.android.domain.Questions.<init>(Questions.java:18)
            at com.firext.android.domain.Questions$1.createFromParcel(Questions.java:22)
            at com.firext.android.domain.Questions$1.createFromParcel(Questions.java:20)
            at android.os.Parcel.readParcelable(Parcel.java:2160)
            at android.os.Parcel.readValue(Parcel.java:2066)
            at android.os.Parcel.readArrayMapInternal(Parcel.java:2393)
            at android.os.BaseBundle.unparcel(BaseBundle.java:221)
            at android.os.Bundle.getParcelable(Bundle.java:738)
            at com.firext.android.activities.reply.ReplyActivity.getData(ReplyActivity.java:72)
            at com.firext.android.activities.reply.ReplyActivity.onCreate(ReplyActivity.java:38)
            at android.app.Activity.performCreate(Activity.java:5720)
            at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1102)
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2208)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2317)
            at android.app.ActivityThread.access$800(ActivityThread.java:143)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1258)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5070)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:836)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:631)
Run Code Online (Sandbox Code Playgroud)

怎么了?

Ric*_*ier 8

摘要

看起来您在存储和检索包含的信息方面存在错误QuestionsGroup.第53行是要注意的路线.

在写入包裹时,Android会为每个字段存储一个特殊的"幻数",以确保在从包裹中提取字段时使用正确的顺序(请参阅数字定义的详细说明).

您收到错误是因为在预期位置找不到幻数 - 这是因为您没有以正确的顺序从包裹中取出字段.


官方源代码的详细说明

[下面的解释是基于"最新"(在撰写本文时)API21的实现BundleBaseBundle类.你可以在这里找到完整的代码:

您还可以使用Android SDK Manager将代码的本地副本安装到开发计算机上,我发现这非常有用.]

所以...

编写parcel时,Android会采取一些安全措施来确保在您尝试取消选中时将字段排成一行.为此,Android会在字段中添加"幻数",定义为:

static final int BUNDLE_MAGIC = 0x4C444E42; // 'B' 'N' 'D' 'L'
Run Code Online (Sandbox Code Playgroud)

你可以在代码中看到这个writeToParcelInner()(总结如下)

void writeToParcelInner(Parcel parcel, int flags) {
    if (mParcelledData != null) {
        if (mParcelledData == EMPTY_PARCEL) {
            parcel.writeInt(0);
        } else {
            int length = mParcelledData.dataSize();
            parcel.writeInt(length);
            parcel.writeInt(BUNDLE_MAGIC);
            parcel.appendFrom(mParcelledData, 0, length);
        }
    } else {

   .......... extra code chopped out for illustration purposes

    }
}
Run Code Online (Sandbox Code Playgroud)

同样,从包裹中读取时,Android会BUNDLE_MAGIC在从包裹返回值之前检查该数字.

private void readFromParcelInner(Parcel parcel, int length) {
    if (length == 0) {
        // Empty Bundle or end of data.
        mParcelledData = EMPTY_PARCEL;
        return;
    }
    int magic = parcel.readInt();
    if (magic != BUNDLE_MAGIC) {
        //noinspection ThrowableInstanceNeverThrown
        throw new IllegalStateException("Bad magic number for Bundle: 0x"
                + Integer.toHexString(magic));
    }

   .......... extra code chopped out for illustration purposes

}
Run Code Online (Sandbox Code Playgroud)

在上面的API21 BaseBundle代码中,您可以看到根据logcat跟踪在1342行引发的错误.


解决方法和进一步调试

原始海报陈述QuestionsGroup正确可分辨.

进一步调试此问题的一种方法是远离(当前)引入BaseBundle该类的新API21 ; 例如,只使用Bundle该类的API19 .

如果用户的代码实际上是正确的,那么它更有可能在更成熟的API上工作.

如果代码仍然中断,则可能是QuestionsGroup代码.但是,此外,新的logcat跟踪将指向代码的不同部分,可以查看并可以提供对问题的更多洞察.

除非是方法中的第1730行Bundle,readFromParcelInner()这意味着同样的问题:

    int magic = parcel.readInt();
    if (magic != BUNDLE_MAGIC) {
        //noinspection ThrowableInstanceNeverThrown
        throw new IllegalStateException("Bad magic number for Bundle: 0x"
                + Integer.toHexString(magic));
    }
Run Code Online (Sandbox Code Playgroud)

相关文章

有关相关信息,请参阅以下帖子:


Bla*_*ama 2

老实说,我从来没有犯过这样的错误。但我使用不同的方式将listof parcelable( QuestionsGroup) 存储在我的parcelable( Questions) 中:

   private Questions(Parcel in) {
        this();
        this.idReport = in.readInt();
        this.nameReport = in.readString();
        this.replyGroups = in.readString();
        //Bundle b = in.readBundle(QuestionsGroup.class.getClassLoader());
        //this.questionsGroup = b.getParcelableArrayList("questionGroups");
        this.questionsGroup = in.readTypedList(questionsGroup,QuestionsGroup.CREATOR);
        this.canCreateNew = (Boolean) in.readValue(Boolean.class.getClassLoader());
    }
Run Code Online (Sandbox Code Playgroud)

和 :

@Override
public void writeToParcel(Parcel dest, int flags) {
    dest.writeInt(this.idReport);
    dest.writeString(this.nameReport);
    dest.writeString(this.replyGroups);
    //Bundle b = new Bundle();
    //b.putParcelableArrayList("questionGroups", (ArrayList<QuestionsGroup>) this.questionsGroup);
    //dest.writeBundle(b);
    dest.writeTypedList(questionsGroup);
    dest.writeValue(this.canCreateNew);
}
Run Code Online (Sandbox Code Playgroud)

我在我的项目中使用上面的代码并且工作正常。您可能想尝试一下,因为它既简单又快速。