当我使用Parcelable接口来读取/写入Boolean时,为什么会发生空对象引用?

4

我正在尝试使一个对象Parcelable,以便传递给我要编写的活动。我实现了Parcelable接口,并编写了一个User类。

User类中有一个布尔属性,所以我参考了答案。即如何在实现Parcelable接口时读取/写入布尔值?

但是出现了一些问题,我收到了一个空引用错误。它说:

java.lang.NullPointerException: Attempt to invoke virtual method 'boolean java.lang.Boolean.booleanValue()' on a null object reference.

以下是代码:

public class User implements Parcelable{
    private String userName;
    private String passWord;
    private Boolean oldUser;

    public String getUserName() {
        return userName;
    }

    public String getPassWord() {
        return passWord;
    }

    public Boolean getOldUser() {
        return oldUser;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public void setPassWord(String passWord) {
        this.passWord = passWord;
    }

    public void setOldUser(Boolean oldUser) {
        this.oldUser = oldUser;
    }

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

    @Override
    public void writeToParcel(Parcel parcel, int i) {
        User user = new User();
        parcel.writeString(user.userName);
        parcel.writeString(user.passWord);
        //seems a good way by using the official api.
        //parcel.writeBooleanArray(new boolean[]{user.oldUser});
        //null pointer,why?
        parcel.writeByte((byte) (user.oldUser ? 1 : 0));
    }

    public static final Parcelable.Creator<User> CREATOR = new Parcelable.Creator<User>() {
        public User createFromParcel(Parcel in) {
            User user = new User();
            //boolean[] myBooleanArr = new boolean[1];
            user.userName = in.readString();
            user.passWord = in.readString();
            //more codes needed to use the below api.
            //in.readBooleanArray(myBooleanArr);
            //user.oldUser = myBooleanArr[0];

            //method we define using readInt.
            user.oldUser = (in.readInt() != 0);
            return user;
        }

        public User[] newArray(int size) {

            return new User[size];
        }
    };
}

日志信息:

日志信息

E/AndroidRuntime: FATAL EXCEPTION: main
              Process: com.example.thon.scos, PID: 2838
              java.lang.NullPointerException: Attempt to invoke virtual method 'boolean java.lang.Boolean.booleanValue()' on a null object reference
                  at es.source.code.model.User$override.writeToParcel(User.java:52)
                  at es.source.code.model.User$override.access$dispatch(User.java)
                  at es.source.code.model.User.writeToParcel(User.java:0)
                  at android.os.Parcel.writeParcelable(Parcel.java:1416)
                  at android.os.Parcel.writeValue(Parcel.java:1322)
                  at android.os.Parcel.writeArrayMapInternal(Parcel.java:665)
                  at android.os.BaseBundle.writeToParcelInner(BaseBundle.java:1330)
                  at android.os.Bundle.writeToParcel(Bundle.java:1079)
                  at android.os.Parcel.writeBundle(Parcel.java:690)
                  at android.content.Intent.writeToParcel(Intent.java:7793)
                  at android.app.ActivityManagerProxy.startActivity(ActivityManagerNative.java:2639)
                  at android.app.Instrumentation.execStartActivity(Instrumentation.java:1507)
                  at android.app.Activity.startActivityForResult(Activity.java:3917)
                  at android.app.Activity.startActivityForResult(Activity.java:3877)
                  at android.support.v4.app.FragmentActivity.startActivityForResult(FragmentActivity.java:861)
                  at android.app.Activity.startActivity(Activity.java:4200)
                  at android.app.Activity.startActivity(Activity.java:4168)
                  at es.source.code.activity.LoginOrRegister$1.onClick(LoginOrRegister.java:46)
                  at android.view.View.performClick(View.java:5198)
                  at android.view.View$PerformClick.run(View.java:21147)
                  at android.os.Handler.handleCallback(Handler.java:739)
                  at android.os.Handler.dispatchMessage(Handler.java:95)
                  at android.os.Looper.loop(Looper.java:148)
                  at android.app.ActivityThread.main(ActivityThread.java:5417)
                  at java.lang.reflect.Method.invoke(Native Method)
                  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
                  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)

诚实地说,我对于Android和Java都是新手。我试着通过编辑

parcel.writeByte((byte) (user.oldUser ? 1 : 0));

为了

parcel.writeByte((byte) (oldUser ? 1 : 0));

太棒了,它确实起作用了,所有的错误都消失了。

这个解决方案正确吗?我就是想不出来。 处理这个问题的最佳方法是什么?谢谢。

2个回答

7

在编写包裹时,不应创建新的User对象。您正在操作当前对象实例。

我猜您可以在createFromParcel()方法中执行所有创建对象和读取包裹的逻辑,但我经常看到下面的模式,其中将包裹传递到对象的构造函数中并在那里处理它。确保按完全相同的顺序读取和写入字段到包裹中。

public class User implements Parcelable {

    private String userName;
    private String passWord;
    private boolean oldUser;

    public User(Parcel in) {
        userName = in.readString();
        passWord = in.readString();
        oldUser = in.readInt() == 1;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(userName);
        dest.writeString(passWord);
        dest.writeInt(oldUser ? 1 : 0);
    }

    public String getUserName() {
        return userName;
    }

    public String getPassWord() {
        return passWord;
    }

    public boolean getOldUser() {
        return oldUser;
    }

    public void setUserName(String userName) {
        this.userName = userName;
    }

    public void setPassWord(String passWord) {
        this.passWord = passWord;
    }

    public void setOldUser(boolean oldUser) {
        this.oldUser = oldUser;
    }

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

    public static final Parcelable.Creator<User> CREATOR = new Parcelable.Creator<User>() {
        public User createFromParcel(Parcel in) {
            return new User(in);
        }

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

}

是的,在createFromParcel()方法中它表现良好,并且在我删除创建用户对象的过程后它仍然有效。坦白地说,我仍然不知道原因,可能只是关于Java语言,而不是Android。我认为我应该处理基本的Java语法。 - xjas
关于构造函数,我必须承认这确实是一种更好的方法。现在适当的方式是只保留样式以防过多的更改。完成演示后,我将根据您的解决方案重构我的代码。谢谢。 - xjas
如果这篇回答对您有帮助,请将其标记为答案。 - Abtin Gramian

4

对于真正的 Boolean(而不是小写的 boolean),我会选择:

@Override
    public void writeToParcel(Parcel out, int flags) {
        if (open_now == null) {
            out.writeInt(-1);
        } else {
            out.writeInt(open_now ? 1 : 0);    
        }

并且

private MyClass(Parcel in) {
    switch (in.readInt()) {
        case 0:
            open_now = false;
            break;
        case 1:
            open_now = true;
            break;
        default:
            open_now = null;
            break;
    }

这将帮助您正确地保留“null”值。


关于如何处理Long类型的任何想法吗? - Ehtesham Hasan
我看到两种方法。1. 保留一些常数来表示null 2. 使用附加字段。 - Vadim

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接