4

我有一个通过两个aidl 实现与自定义服务交互的应用程序。运行服务时,我遇到了以下异常:

ERROR/AndroidRuntime(9435): FATAL EXCEPTION: main
ERROR/AndroidRuntime(9435): java.lang.NullPointerException
ERROR/AndroidRuntime(9435): at android.os.Parcel.readException(Parcel.java:1328)
ERROR/AndroidRuntime(9435): at android.os.Parcel.readException(Parcel.java:1276)
ERROR/AndroidRuntime(9435): at myservice.IAsyncService$Stub$Proxy.addItems(IAsyncService.java:259)

从这个错误来看,我认为这与执行的编组有关。所以我检查了两个aidl实现;急救(使用原语)似乎工作正常(我认为因为他们不需要我指定导入/实现);但是,第二个(使用自定义对象)似乎崩溃了,给出了上述错误。我已确保自定义对象正确实现了 parcelable(请参阅帖子底部的实现),甚至尝试从单独的aidl 文件中引用自定义对象,如下所示。

(单独的aidl文件。)-(IMyItem.aidl)

package myservice;
parcelable myservice.MyItem;

(其他辅助文件。) - (IAsyncService.aidl)

package myservice;
import myservice.IMyItem;

interface IAsyncService
{
    void addItems(in List<MyItem> itemCollection);
}

(Parcelable 实现)(MyItem)

public class MyItem implements Parcelable
{
    private String name = "";

    public MyItem(String name)
    {
        this.name = name.trim();
    }

    public MyItem(Parcel source) {
        readFromParcel(source);
    }

    public void setItem(String name)
    {
        this.name = name;
    }

    public String getName()
    {
        return this.name;
    }

    public static final Parcelable.Creator<MyItem> CREATOR = new Parcelable.Creator<MyItem>() {
        @Override
        public MyItem[] newArray(int size) {
            return new MyItem[size];
        }

        @Override
        public MyItem createFromParcel(Parcel source) {
            return new MyItem(source);
        }
    };

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

    @Override
    public void writeToParcel(Parcel destination, int flags) {
        destination.writeString(this.name);
    }

    public void readFromParcel(Parcel source) {
        this.name = source.readString();
    }
}

有什么我遗漏或完全忽略的吗?

谢谢约翰

4

1 回答 1

8

您看到的是 NullPointerException 已在远程端引发,该异常作为结果返回,并重新引发给您的客户端。在这些情况下,它有助于调试找出原始异常是什么。你可以用一个技巧来做到这一点:

在实现接口的类中,重写 onTransact() 以执行此操作:

protected boolean onTransact(int code, Parcel data, Parcel reply, int flags) {
    try {
        super.onTransact(code, data, reply, flags);
    } catch (RuntimeException e) {
        Log.w("MyClass", "Unexpected remote exception", e);
        throw e;
    }
}

现在,当崩溃发生时,您还将在日志中看到打印的原始异常,以及它发生的行。

于 2011-06-22T16:22:42.013 回答