【问题标题】:Android: "Class loader may fail for processes that host multiple applications"Android:“托管多个应用程序的进程的类加载器可能会失败”
【发布时间】:2016-08-30 19:08:50
【问题描述】:

Eclipse 的 Android 的 logcat 中的这条消息是什么意思?

W/ActivityThread: ClassLoader.getResources: The class loader returned by Thread.getContextClassLoader() may fail for processes that host multiple applications. You should explicitly specify a context class loader. For example: Thread.setContextClassLoader(getClass().getClassLoader());

不幸的是,没有给出关于这个警告的上下文,所以我不知道是什么导致了这个问题以及如何解决它。

【问题讨论】:

  • 所以您没有在代码中的任何地方调用 Thread.getContextClassLoader() 吗?也许你正在使用的图书馆?
  • 不,甚至不是这个。我的项目及其库中从未调用过getContextClassLoader()

标签: android multithreading classloader


【解决方案1】:

背景资料

该消息表示 Android 已使用 Thread.currentThread().setContextClassLoader() 设置了一个虚拟的 ClassLoader,并且尝试使用该虚拟类加载器。 something 可以是很多东西,很难从给出的信息中准确地说出什么。不过,您可以尝试一个技巧,请参见下文。无论如何,当进程可能包含来自多个 APK 的代码的风险时,Android 会设置虚拟类加载器。更具体地说,如果您使用了android:sharedUserId,Android 会查看您的清单:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    ...
    android:sharedUserId="triggers.dummy.loader" >

或者如果你运行在非标准的android:process

    <application android:process="triggers.dummy.loader">

如何摆脱警告

您可以做两件事来消除警告:

  1. 不要使用android:sharedUserIdandroid:process
  2. 在运行任何其他代码之前明确设置要使用的 APK ClassLoader

要采用解决方案 2,您需要一些关键的见解。首先,对于 APK 中的任何类 AnyClassAnyClass.class.getClassLoader() 将返回相同的 ClassLoader。二、

AnyClass obj = new AnyClass();
Thread.currentThread().setContextClassLoader(obj.getClass().getClassLoader())

相同
Thread.currentThread().setContextClassLoader(AnyClass.class.getClassLoader())

第三,你需要在调用Thread.currentThread().getContextClassLoader()的代码之前调用Thread.currentThread().setContextClassLoader(getClass().getClassLoader())。 第四,当涉及到多个 APK 时,需要在最后一个 APK 加载完成后调用Thread.setContextClassLoader(getClass().getClassLoader())(否则加载最后一个 APK 会覆盖你手动设置的内容)。因此,最好使用下面的调试技巧找出谁在使用上下文类加载器。然后,在此之前,您为来自所需 APK 的类调用 Thread.setContextClassLoader(getClass().getClassLoader()),通常是首先加载的 APK(或者,在仅涉及一个 APK 的情况下,该 APK ;)。第五,上下文类加载器是per thread,如果你的应用程序是多线程的,你需要牢记这一点。

调试技巧

如果您想找出调用 ClassLoader.getResources() 的代码,这应该可以:

Thread.currentThread().setContextClassLoader(new ClassLoader() {
    @Override
    public Enumeration<URL> getResources(String resName) throws IOException {
        Log.i("Debug", "Stack trace of who uses " +
                "Thread.currentThread().getContextClassLoader()." +
                "getResources(String resName):", new Exception());
        return super.getResources(resName);
    }
});

如果您尽早执行此操作,您应该会在 logcat 中看到一个堆栈跟踪,该堆栈跟踪可以追溯到在虚拟类加载器上调用 getResources() 的任何人。

【讨论】:

  • 非常感谢!我已经设置了sharedUserId,所以我们已经发现了问题:) 但是你能说,只要我不真正使用sharedUserId,一切都很好吗?我已经设置了,因为没有它的版本发布后无法设置。
  • 对,只要你不需要sharedUserId,就不应该设置。如果在发布后设置,可能会遇到应用程序数据迁移问题,但如果提前设置,则可能会遇到应用程序数据迁移问题,因为您不知道要设置什么值。
  • 当然,您可以将其设置为您的包名;)然后,当您在另一个 apk 中需要它时,也将其设置为该包名。这样,您将不会遇到任何数据迁移问题。问题是:您不仅有一些迁移问题,还有严重的异常(应用程序无法再找到自己的数据)。我只是问:如果我设置了它但不使用它,我可以忽略类加载器警告吗?
  • 啊哈,你是这个意思。好吧,您会收到警告,因为有些东西正在使用getContextClassLoader()。如果某些东西需要访问您的 APK 特定类和资源,它目前无法访问这些。即使当前不需要访问您的 APK 特定类和资源,这种需求也可能在未来出现。因此,如果您想继续使用sharedUserId,我建议您选择解决方案 2,即使现在可能不需要它,因为将来可能需要它(当您忘记此警告时)。
  • 谢谢!但是如果还不需要外部访问我的 apk,我该如何使用setContextClassLoader?只有当我有第二个具有相同sharedUserId 的想要访问我的第一个 APK 的 APK 时,我才能使用它,对吧?以及如何使用它?只需在访问其他 APK 方法的代码之前调用 Thread.setContextClassLoader(getClass().getClassLoader()) 即可?
【解决方案2】:

我收到此警告,但我的清单中没有 android:sharedUserId 或 android:process...

发现它只在模拟器上呈现...

设备未显示警告消息。在智能手机 KitKat 4.4 API 19 和平板电脑 5.0.1 API 21 上测试。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2010-11-08
    • 1970-01-01
    • 2020-11-23
    • 2018-11-15
    • 1970-01-01
    • 2014-04-20
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多