21

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());

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

4

2 回答 2

35

背景资料

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

<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. ClassLoader在运行任何其他代码之前明确设置要使用的 APK

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

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

是相同的

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

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

调试技巧

如果您想找出调用 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()虚拟类加载器的任何人。

于 2012-11-19T13:48:53.140 回答
0

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

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

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

于 2016-08-30T11:08:50.473 回答