Android 服务无法启动,因为 class 的 ClassNotFoundException 甚至不再存在

Android Service cannot start because of a ClassNotFoundException from a class that doesn't even exist anymore

我的 activity 'used' 在启动 Service 时将自定义(可序列化)class 放入包中。但是我已经删除了它(并删除了 class)并直接将基元放入包中。

现在...我的 ACRA 日志完全被 onStartCommand 在线读取包时崩溃的错误所困扰。

java.lang.RuntimeException: Unable to start service com.xxx.MyService@db4c909 with Intent { act=ACTION_START flg=0x4 cmp=com.xxx/.MyService (has extras) }: java.lang.RuntimeException: Parcelable encountered ClassNotFoundException reading a Serializable object (name = com.xxx.MyBundle)
at android.app.ActivityThread.handleServiceArgs(ActivityThread.java:3045)
at android.app.ActivityThread.access00(ActivityThread.java:157)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1454)
at android.os.Handler.dispatchMessage(Handler.java:102)
at android.os.Looper.loop(Looper.java:148)
at android.app.ActivityThread.main(ActivityThread.java:5551)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:730)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:620)
Caused by: java.lang.RuntimeException: Parcelable encountered ClassNotFoundException reading a Serializable object (name = com.xxx.MyBundle)
at android.os.Parcel.readSerializable(Parcel.java:2491)
at android.os.Parcel.readValue(Parcel.java:2294)
at android.os.Parcel.readArrayMapInternal(Parcel.java:2592)
at android.os.BaseBundle.unparcel(BaseBundle.java:221)
at android.os.BaseBundle.get(BaseBundle.java:281)
at com.xxx.MyService.onStartCommand(MyService.java:190)
at android.app.ActivityThread.handleServiceArgs(ActivityThread.java:3028)
....
... 14 more
Caused by: java.lang.ClassNotFoundException: Didn't find class "com.xxx.MyBundle" on path: DexPathList[[zip file "/data/app/com.xxx-2/base.apk"],nativeLibraryDirectories=[/data/app/com.xxx-2/lib/arm, /data/app/com.xxx-2/base.apk!/lib/armeabi-v7a, /vendor/lib, /system/lib]]
at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56)
at java.lang.ClassLoader.loadClass(ClassLoader.java:511)
at java.lang.ClassLoader.loadClass(ClassLoader.java:469)

我不知道 com.xxx.MyBundle 甚至不存在于项目中!

请注意,我在崩溃日志中看到了这个错误。我无法在本地重现这个,可能是因为我已经多次卸载并重新安装该应用程序,这可能会解决问题。

这里的问题是应用程序可能崩溃的人有数千人,不能只告诉所有人 "uninstall- reinstall"。我想找到这个问题的原因,并可能通过新的更新来修复它,而无需用户手动卸载应用程序并重新安装..

编辑

这是事件的历史

所以你看..这不是服务 运行 与否的问题..即使我的用户从 v3 更新到 v4 我也希望这个错误消失但是没有...我仍然看到 v4

中的 ClassNotFoundExcpetionn

正如我从您的问题中猜测的那样,您可能已经使用您的应用程序的先前版本启动了 Service,该应用程序最近与所使用的 classes 一起被删除。它导致 ClassNotFoundException

因此,在这种特定情况下,您可以考虑检测应用程序更新,并且在应用程序更新时您需要检查 Service 是否在后台 运行 需要停止。

要检查是否安装了更新版本的应用程序,您将收到一个名为 ACTION_MY_PACKAGE_REPLACED 的广播事件,然后您停止 运行 Service

要检查您想要的 Service 是否已经 运行,您可以考虑使用这样的检查器。

private boolean isMyServiceRunning(Class<?> serviceClass) {
    ActivityManager manager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
    for (RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
        if (serviceClass.getName().equals(service.service.getClassName())) {
            return true;
        }
    }
    return false;
}

然后使用

调用它
if(isMyServiceRunning(MyService.class))
    stopMyService();

更新

我想我已经正确理解了你的问题并且已经想到了上面的解决方案。我猜,您已经在 v1 或 v2 中启动的 Service 是 运行 并期待那里的序列化包 class。因此,您需要停止该服务并重新启动它以使其遵循您期望的当前行为。