3

在 Android 上,我们有一个类,它包装了来自 GMS(谷歌移动服务)的 LocationClient 对象。(注意 LocationClient 实现了 com.google.android.gms.common.GooglePlayServicesClient)。

不幸的是,LocationClient 对象有抛出 DeadObjectExceptions 的习惯(例如,当我们调用locationClient.getLastLocation()时),我们通过我们的几种日志记录机制检测到了这种情况。然而,奇怪的是,LocationClient 没有被记录为抛出 DeadObjectExceptions,而且我只能在它们发生 o_0 的时间的 1/40 时捕捉到所述 DeadObjectExceptions。我们对这个问题没有重现,我个人从未见过它,但是它发生在我们的大量用户身上。

其他注意事项:

[a] “Caused by: java.lang.IllegalStateException: android.os.DeadObjectException”这一行是关于什么的?这两种异常类型没有祖先-后代关系

[b] 我在 Android 论坛上发帖,但他们当然拒绝了我的帖子,认为是“错误的论坛”,而且没有 GMS 论坛,所以我完全不走运。

总之,问题是:GMS 正在触发这个奇怪的无法捕获的异常,那么这是怎么回事,我该怎么办?

Here's a stack trace:
com.myapp.android.service.AsyncExecutionException
     at com.myapp.android.service.AsyncService$ExceptionThrower.run(MyApp:120)
     at android.os.Handler.handleCallback(Handler.java:615)
     at android.os.Handler.dispatchMessage(Handler.java:92)
     at android.os.Looper.loop(Looper.java:137)
     at android.app.ActivityThread.main(ActivityThread.java:4794)
     at java.lang.reflect.Method.invokeNative(Method.java)
     at java.lang.reflect.Method.invoke(Method.java:511)
     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:789)
     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:556)
     at dalvik.system.NativeStart.main(NativeStart.java)
Caused by: java.lang.IllegalStateException: android.os.DeadObjectException
     at com.google.android.gms.internal.ey.getLastLocation()
     at com.google.android.gms.internal.ez.getLastLocation()
     at com.google.android.gms.location.LocationClient.getLastLocation()
     ***at com.myapp.GoogleLocationProvider.getLastLocation(MyApp:92)***
     at com.myapp.LocationProducer.getLocation(MyApp:183)
     at com.myapp.LocationProducer.getLocationHeader(MyApp:194)
     at com.myapp.API.onExecute(MyApp:344)
     ...
     at java.lang.Thread.run(Thread.java:856)
Caused by: android.os.DeadObjectException
     at android.os.BinderProxy.transact(Binder.java)
     at com.google.android.gms.internal.ex$a$a.a()
     at com.google.android.gms.internal.ey.getLastLocation()
     at com.google.android.gms.internal.ez.getLastLocation()
     ***at com.google.android.gms.location.LocationClient.getLastLocation()***
     at com.myapp.GoogleLocationProvider.getLastLocation(MyApp:92)
     at com.myapp.LocationProducer.getLocation(MyApp:183)
     at com.myapp.LocationProducer.getLocationHeader(MyApp:194)
     ...
     at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
     at java.util.concurrent.FutureTask.run(FutureTask.java:137)
     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
     at java.lang.Thread.run(Thread.java:856)

------------ ADDENDUM ------------- 这是我们的实际代码。你会注意到我们总是事先检查 mLocationClient.isConnected(),所以这不是问题。有可能我们变得非常不走运,并且 mLocationObject 在调用 isOnConnected() 和 getLastLocation() 之间死亡,但这对我来说似乎不太可能。我想我可以在通话之前、通话之间和通话后开始记录并找出答案。

LocationClient mLocationClient; // populated somewhere

public Location getLastLocation() {
    if (!mLocationClient.isConnected()) {
        return null;
    }
    Location location = null;
    try {
        location = mLocationClient.getLastLocation();
    } catch (Exception e) {
        if (!handleDeadObjectException(e)) {
            throw e;
        }
    }
    return location;
}

// logs, attempts to handle depending on user configuration
private boolean handleDeadObjectException(Exception e);
4

1 回答 1

9

从文档DeadObjectException

您正在调用的对象已经死亡,因为它的托管进程不再存在。

这意味着,您正试图在不再可用的不同进程中访问一个对象。例如,如果您绑定到在不同进程中运行的服务(即 Google 移动服务),那么您使用的IBinder是一个本地对象,它“代表”远程进程中的一个对象。当远程对象不再可用,并且您尝试使用本地 IBinder 对象时,您将收到DeadObjectException

所以...

a] 什么是“Caused by: java.lang.IllegalStateException: android.os.DeadObjectException”这一行?这两种异常类型没有祖先-后代关系

这两个例外没有任何联系。IllegalStateException是实际异常,而DeadObjectException根异常。

由于gms.location.LocationClient.getLastLocation()不想声明暴露内部实现的 throw 元素 - 使用活页夹等 - 它根本不想。但是当发生诸如DeadObjectException之类的异常时,它仍然想抛出,因此它使用运行时异常IllegalStateException(不需要抛出声明)。

[b] 我在 Android 论坛上发帖,但他们当然拒绝了我的帖子,认为是“错误的论坛”,而且没有 GMS 论坛,所以我完全不走运。

:(

总之,问题是:GMS 正在触发这个奇怪的无法捕获的异常,那么这是怎么回事,我该怎么办?

使用 GMS LocationClient 时,您需要在与客户端交互之前检查LocationClient.isConnected()是否。请注意,有时LocationClient.isConnected()会返回 true,但在调用LocationClient.getLastLocation()之后可能仍会抛出java.lang.IllegalStateException: android.os.DeadObjectException,其原因是线程问题和客户端连接的竞争条件当你检查但在你实际行动之前连接丢失了。

您应该做的是 a) 检查客户端是否已连接

if ( mLocationClient != null && mLocationClient.isConnected() ) {   
    mLocationClient.getLastLocation();
}

b) 捕获IllegalStateException(而不是DeadObjectException

if ( mLocationClient != null && mLocationClient.isConnected() ) {   
    try {
        mLocationClient.getLastLocation();
    } catch (IllegalStateException ex) {
        // This will catch the exception, handle as needed
    }
}
于 2014-07-06T09:22:09.817 回答