Android服务无法启动,因为来自不再存在的类的ClassNotFoundException

时间:2017-01-14 16:36:54

标签: android serialization service bundle classnotfoundexception

我的活动'用过'在启动Service时,在bundle中放置一个自定义(可序列化)类。但是夫妻版本后面我删除了它(并删除了类)并直接将原语放在包中。

现在......我的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.access$2200(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在项目中甚至不再存在,我不会得到错误!

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

这里的问题是成千上万的人可能会崩溃并且不能告诉所有人"卸载 - 重新安装"。我想找到原因,并可能使用新的更新修复它,而无需用户手动卸载应用程序并重新安装..

修改

这是事件的历史

  • v1:启动服务时,我在bundle中放了一个Serializable类    (我忘了设置serialVersionUID)我在启动服务时看到了很多序列化错误,例如在进行干接时版本不匹配时...忘记serialVersionUID时的典型错误

  • v2:我在Serializable类中添加了serialVersionUID = -1    (没有骰子,在日志中我仍然看到这些序列化现在说发现版本= {随机数}但是-1 ....

  • v3:我放弃并删除了com.xx.MyBundle并创建了一个com.xx.MyBundle2    (没有Dice ..这就是我开始在日志中看到ClassNotFoundException)

  • v4:我删除com.xx.MyBundle2而不是将可序列化的类放在我直接放入原语的包中

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

1 个答案:

答案 0 :(得分:2)

正如我从您的问题中猜测的那样,您可能已经开始使用应用程序的先前版本的Service,该版本最近已与所使用的类一起删除。它导致了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正在运行,并期待那里的序列化类包。因此,您需要停止服务并再次启动它,使其遵循您期望的当前行为。