我正在开发一个Android应用,我们正在调查内存使用情况。
查看来自hprof的堆转储,我们发现在JarURLConnectionImpl的静态缓存中使用了将近2M(占堆的22%):
查看source code for JarURLConnectionImpl,似乎条目已添加到静态jarCache变量,但从未删除。
如果确实没有删除它们,那将是潜在的内存泄漏。
这是泄漏吗?是否有修复程序或解决方法?
最佳答案
这是一个丑陋的解决方法:
private static HashMap<URL,JarFile> jarCache;
static {
try {
Class<?> jarURLConnectionImplClass = Class.forName("org.apache.harmony.luni.internal.net.www.protocol.jar.JarURLConnectionImpl");
final Field jarCacheField = jarURLConnectionImplClass.getDeclaredField("jarCache");
jarCacheField.setAccessible(true);
//noinspection unchecked
jarCache = (HashMap<URL, JarFile>) jarCacheField.get(null);
} catch(Exception e) {
// ignored
}
}
然后,定期运行以下命令:
// HACK http://stackoverflow.com/questions/14610350/android-memory-leak-in-apache-harmonys-jarurlconnectionimpl
if( jarCache!=null ) {
try {
for (
final Iterator<Map.Entry<URL, JarFile>> iterator = jarCache.entrySet().iterator(); iterator.hasNext(); ) {
final Map.Entry<URL, JarFile> e = iterator.next();
final URL url = e.getKey();
if (Strings.toString(url).endsWith(".apk")) {
Log.i(TAG,"Removing static hashmap entry for " + url);
try {
final JarFile jarFile = e.getValue();
jarFile.close();
iterator.remove();
} catch( Exception f ) {
Log.e(TAG,"Error removing hashmap entry for "+ url,f);
}
}
}
} catch( Exception e ) {
// ignored
}
}
我在 Activity 创建时运行它,因此每次创建我的 Activity 之一时都会执行它。丑陋的hashmap条目似乎并没有那么频繁地被重新创建,但是它似乎偶尔会重新出现,因此仅运行一次此代码是不够的。