Bitmap的生成和释放流程

BitmapFactory.decodeResource和BitmapFactory.decodeStream,相信对于有过android app开发经验的人来说都是很熟悉了。关于Bitmap的OOM问题,网上也有很多文章进行了分析,不少文章都说为避免OOM,最好使用BitmapFactory.decodeStream,但是具体说明原因的我至今没有找到,所以趁着10.1期间有空,就调查了一番,希望结果对大家能有帮助。

      先来张时序图,看了这张图,估计很多问题都不用说明了:


 

下面做些说明:

1,函数externalBytesAvailable(...)的内存计算方法详细请参见文章《Android Bitmap内存限制》;

2,Bitmap.createScaledBitmap时可能会对Bitmap进行缩放,缩放所使用的Options请参见下面这段代码:

 

  1. public static Bitmap decodeResourceStream(Resources res, TypedValue value,  
  2.         InputStream is, Rect pad, Options opts) {  
  3.   
  4.     if (opts == null) {  
  5.         opts = new Options();  
  6.     }  
  7.   
  8.     if (opts.inDensity == 0 && value != null) {  
  9.         final int density = value.density;  
  10.         if (density == TypedValue.DENSITY_DEFAULT) {  
  11.             opts.inDensity = DisplayMetrics.DENSITY_DEFAULT;  
  12.         } else if (density != TypedValue.DENSITY_NONE) {  
  13.             opts.inDensity = density;  
  14.         }  
  15.     }  
  16.       
  17.     if (opts.inTargetDensity == 0 && res != null) {  
  18.         opts.inTargetDensity = res.getDisplayMetrics().densityDpi;  
  19.     }  
  20.       
  21.     return decodeStream(is, pad, opts);  
  22. }  
当放大时,会加大造成OOM的可能性;
3,BitmapFactory.decodeResource比起BitmapFactory.decodeStream来说,在没有参数Options opts的情况下,确实是多占用了内存,因为多了BitmapFactory.finishDecode这一步,这里多了一个Bitmap.createScaledBitmap操作;

 

4,关于使用BitmapFactory.decodeResource和BitmapFactory.decodeStream的效率对比,大家可以参见文章《Android 内存优化测试

5,关于文章《Android 内存优化测试》提到的使用BitmapDrawable是最节省内存的方法,原因是frameworks\base\core\java\android\content\res\Resources.java对Bitmap做了缓存处理,文中所有的array其实都是指向同一个Bitmap,这一点也可以通过Bitmap.toString的输出信息来证明。

备注:时序图看不清楚的,可以到http://download.csdn.net/detail/imyfriend/4617329下载原图。

 

 

 
 

上篇文章谈到了Bitmap的生成,现在说说Bitmap的释放,要不总是有种没说完的感觉。

      Bitmap的释放相对来说,比较简单,还是先上张时序图吧!

 

一些说明:
1,AndroidPixelRef的设置是在Bitmap生成的GraphicsJNI::setJavaPixelRef里完成的,这里可以参见文章《Bitmap的生成流程之BitmapFactory.decodeResource VS BitmapFactory.decodeStream》。具体代码如下:

  1. bool GraphicsJNI::setJavaPixelRef(JNIEnv* env, SkBitmap* bitmap,  
  2.                                   SkColorTable* ctable, bool reportSizeToVM) {  
  3. ......  
  4.     SkPixelRef* pr = reportSizeToVM ?  
  5.                         new AndroidPixelRef(env, addr, size, ctable) :  
  6.                         new SkMallocPixelRef(addr, size, ctable);  
  7.   
  8.   
  9. ......  
  10. }  
2,VMRuntime.trackExternalFree把Bitmap占用的虚拟机heap内存归还。如果你有兴趣的话,请具体参见dalvik\vm\native\dalvik_system_VMRuntime.c。
posted @ 2012-12-03 01:04  OYK  阅读(1885)  评论(0编辑  收藏  举报