gpt4 book ai didi

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

转载 作者:太空狗 更新时间:2023-10-29 13:10:31 24 4
gpt4 key购买 nike

我的 Activity 在启动 Service 时“使用”了将自定义(可序列化)类放入包中。但是我删除了它(并删除了类)并直接将基元放入包中。

现在......我的 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:我在启动服务时将一个 Serializable 类放入包中 (我忘记设置 serialVersionUID)并且在启动服务时看到很多序列化错误,例如反序列化时版本不匹配。忘记 serialVersionUID 时的典型错误

  • v2:我在我的可序列化类中添加了 serialVersionUID=-1 (没有骰子,在日志中我仍然看到这些序列化现在说 found version={random number} but was -1 ....

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

  • v4:我删除了 com.xx.MyBundle2,而不是将可序列化类放入包中,而是直接放入我的基元

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

最佳答案

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

关于Android 服务无法启动,因为来自一个甚至不再存在的类的 ClassNotFoundException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41652355/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com