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());
不幸的是,没有给出关于这个警告的上下文,所以我不知道是什么导致了这个问题以及如何解决它。
最佳答案
背景资料
该消息表示 Android 已使用 Thread.currentThread().setContextClassLoader()
设置了一个虚拟 ClassLoader
,并且某些东西试图使用该虚拟类加载器。 something 可以是很多东西,很难从给出的信息中准确地说出什么。不过,您可以尝试一个技巧,请参见下文。无论如何,当进程可能包含来自多个 APK 的代码的风险时,Android 会设置虚拟类加载器。更具体地说,如果您使用了 android:sharedUserId
,Android 会查看您的 list :
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
...
android:sharedUserId="triggers.dummy.loader" >
或者如果你在非标准的android:process
<application android:process="triggers.dummy.loader">
如何摆脱警告
您可以做两件事来消除警告:
- 不要使用
android:sharedUserId
或android:process
- 在运行任何其他代码之前明确设置要使用的 APK
ClassLoader
要采用解决方案 2,您需要一些关键的见解。首先,对于 APK 中的任何类 AnyClass
,AnyClass.class.getClassLoader()
将返回相同的 ClassLoader
。二、
AnyClass obj = new AnyClass();
Thread.currentThread().setContextClassLoader(obj.getClass().getClassLoader())
与
相同Thread.currentThread().setContextClassLoader(AnyClass.class.getClassLoader())
三、需要在调用Thread.currentThread().getContextClassLoader()的代码之前调用
。
四、当涉及到多个APK时,需要调用Thread.currentThread().setContextClassLoader(getClass().getClassLoader())
/Thread.setContextClassLoader(getClass().getClassLoader())
在最后一个APK加载完成之后(否则加载最后一个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()
的任何人。
关于安卓: "Class loader may fail for processes that host multiple applications",我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13407006/