Android WebView Memory Leak WebView内存泄漏

移动开发 Android
要使用WebView不造成内存泄漏,首先应该做的就是不能在xml中定义webview节点,而是在需要的时候动态生成。即:可以在使用WebView 的地方放置一个LinearLayout类似ViewGroup的节点,然后在要使用WebView的时候,动态生成。

[[117125]]

在这次开发过程中,需要用到webview展示一些界面,但是加载的页面如果有很多图片就会发现内存占用暴涨,并且在退出该界面后,即使在包含该 webview的Activity的destroy()方法中,使用webview.destroy();webview=null;对内存占回收用还是没有任何效果。有人说,一旦在你的xml布局中引用了webview甚至没有使用过,都会阻碍重新进入Application之后对内存的gc。包括使用 MapView有时一会引发OOM,几经周折在网上看到各种解决办法,在这里跟大家分享一下。但是到目前为止还没有找到根本的解决办法,网上也有说是 sdk的bug。但是不管怎么样,我们还是需要使用的。

要使用WebView不造成内存泄漏,首先应该做的就是不能在xml中定义webview节点,而是在需要的时候动态生成。即:可以在使用WebView 的地方放置一个LinearLayout类似ViewGroup的节点,然后在要使用WebView的时候,动态生成即:

  1. WebView      mWebView = new WebView(getApplicationgContext()); 
  2. LinearLayout mll      = findViewById(R.id.xxx); 
  3. mll.addView(mWebView); 

然后一定要在onDestroy()方法中显式的调用

  1. protected void onDestroy() { 
  2.       super.onDestroy(); 
  3.       mWebView.removeAllViews(); 
  4.       mWebView.destroy() 

注意: new  WebView(getApplicationgContext()) ;必须传入ApplicationContext如果传入Activity的 Context的话,对内存的引用会一直被保持着。有人用这个方法解决了当Activity被消除后依然保持引用的问题。但是你会发现,如果你需要在 WebView中打开链接或者你打开的页面带有flash,获得你的WebView想弹出一个dialog,都会导致从 ApplicationContext到ActivityContext的强制类型转换错误,从而导致你应用崩溃。这是因为在加载flash的时候,系统 会首先把你的WebView作为父控件,然后在该控件上绘制flash,他想找一个Activity的Context来绘制他,但是你传入的是 ApplicationContext。后果,你可以晓得了哈。

于是大牛们就Activity销毁后还保持引用这个问题,提供了另一种解决办法:既然你不能给我删除引用,那么我就自己来吧。于是下面的这种方法诞生了:

(作者说这个方法是依赖android.webkit implementation有可能在最近的版本中失败)

  1. public void setConfigCallback(WindowManager windowManager) { 
  2.     try { 
  3.         Field field = WebView.class.getDeclaredField("mWebViewCore"); 
  4.         field = field.getType().getDeclaredField("mBrowserFrame"); 
  5.         field = field.getType().getDeclaredField("sConfigCallback"); 
  6.         field.setAccessible(true); 
  7.         Object configCallback = field.get(null); 
  8.   
  9.         if (null == configCallback) { 
  10.             return
  11.         } 
  12.   
  13.         field = field.getType().getDeclaredField("mWindowManager"); 
  14.         field.setAccessible(true); 
  15.         field.set(configCallback, windowManager); 
  16.     } catch(Exception e) { 
  17.     } 

然后在Activity中调用上面的方法:

  1. public void onCreate(Bundle savedInstanceState) { 
  2.     super.onCreate(savedInstanceState); 
  3.     setConfigCallback((WindowManager)getApplicationContext().getSystemService(Context.WINDOW_SERVICE)); 
  4.   
  5. public void onDestroy() { 
  6.     setConfigCallback(null); 
  7.     super.onDestroy(); 

该反射方法在我的实验中(2.3.6)确实有些用处,在应用内存占用到70M左右的时候会明显释放到50M或者60M然后的释放就有些缓慢,其实就是看不出来了。之前在没使用该方法的时候可能达到120M。

但是!!!我们的应用要求占用内存更低啊,这肿么拌?凉拌么?No。在各种纠结之后,终于找到了***解决办法!!!该办法适用于我们的需求,在退出 WebView的界面之后,迅速回收内存。要问这个方法是什么,不要9999,不要8999,只要你仔细看好下面一句话:那就是为加载WebView的界 面开启新进程,在该页面退出之后关闭这个进程。

这一点说了之后,你懂了吧?
但是在这个其中,杀死自己进程的时候又遇到了问题,网上介绍的各种方法都不好使,
killBackgroundProcesses(getPackageName());各种不好用,***使用System.exit(0);直接退出虚拟机(Android为每一个进程创建一个虚拟机的)。这个肯定不用纠结了,一旦退出,内存里面释放。听涛哥说QQ也是这么做。

***英雄要问出处,附上大牛解说引起该问题的出处

这个泄漏出现在external/webkit/Source/WebKit/android/WebCoreSupport/UrlInterceptResponse.cpp.中。具体我自己真心没有深入研究。大家有兴趣的话,可以看看哈。

  1. --- a/Source/WebKit/android/WebCoreSupport/UrlInterceptResponse.cpp 
  2. +++ b/Source/WebKit/android/WebCoreSupport/UrlInterceptResponse.cpp 
  3. @@ -63,10 +63,10 @@ public
  4.          JNIEnv* env = JSC::Bindings::getJNIEnv(); 
  5.          // Initialize our read buffer to the capacity of out. 
  6.          if (!m_buffer) { 
  7. -            m_buffer = env->NewByteArray(out->capacity()); 
  8. -            m_buffer = (jbyteArray) env->NewGlobalRef(m_buffer); 
  9. +            ScopedLocalRef<jbyteArray> buffer_local(env, env->NewByteArray(out->capacity())); 
  10. +            m_buffer = static_cast<jbyteArray>(env->NewGlobalRef(buffer_local.get())); 
  11.          } 
  12.          int size = (int) env->CallIntMethod(m_inputStream, m_read, m_buffer); 
  13.          if (checkException(env) || size < 0
  14.              return
  15.          // Copy from m_buffer to out. 

而且从这里https://github.com/android/platform_external_webkit/commit/1e3e46a731730c02d916ea805ec4b20191509282这个bug的解决状态。

还有一个问题要说的,也是在WebView使用的时候出现的问题:WebView 中包含一个ZoomButtonsController,当使用 web.getSettings().setBuiltInZoomControls(true);启用该设置后,用户一旦触摸屏幕,就会出现缩放控制图 标。这个图标过上几秒会自动消失,但在3.0系统以上上,如果图标自动消失前退出当前Activity的话,就会发生ZoomButton找不到依附的 Window而造成程序崩溃,解决办法很简单就是在Activity的ondestory方法中调用 web.setVisibility(View.GONE);方法,手动将其隐藏,就不会崩溃了。在3.0一下系统上不会出现该崩溃问题,真是各种崩溃, 防不胜防啊!

***还有内存泄漏的一些个建议:

In summary, to avoid context-related memory leaks, remember the following:

  • Do not keep long-lived references to a context-activity (a reference to an activity should have the same life cycle as the activity itself)
  • Try using the context-application instead of a context-activity
  • Avoid non-static inner classes in an activity if you don’t control their life cycle, use a static inner class and make a weak reference to the activity inside

And remember that a garbage collector is not an insurance against memory leaks. Last but not least, we try to make such leaks harder to make happen whenever we can.

责任编辑:闫佳明 来源: oschina
相关推荐

2021-12-27 09:33:12

内存泄漏程序

2015-03-03 15:53:31

Android控件

2011-08-19 14:27:29

iPhone开发

2019-05-14 11:21:07

FlutterAndroidWebView

2014-09-17 10:57:30

AndroidWebview安全交互

2014-07-29 11:16:07

2010-08-04 10:17:17

Android开发WebView组件

2013-09-13 13:15:28

AndroidWebViewJavaScript

2023-10-31 16:40:38

LeakCanary内存泄漏

2016-03-21 10:31:25

Android内存泄露

2016-12-22 17:21:11

Android性能优化内存泄漏

2015-01-23 16:32:52

2012-12-26 12:41:14

Android开发WebView

2016-10-24 14:04:24

2013-08-07 10:16:43

Android内存泄漏

2016-07-05 14:09:02

AndroidJAVA内存

2011-06-01 12:50:41

Android 内存

2013-05-17 13:52:00

Android开发Android Web获取WebView宽度

2013-07-03 16:49:17

AndroidWebView

2016-09-14 22:22:03

Android Vue性能优化
点赞
收藏

51CTO技术栈公众号