Android Bitmap压缩策略

1、为何Bitmap须要高效加载?

如今的高清大图,动辄就要好几M,而Android对单个应用所施加的内存限制,只有
小几十M,如16M,这致使加载Bitmap的时候很容易出现内存溢出。以下异常信
息,即是在开发中常常须要的:java

java.lang.OutofMemoryError:bitmap size exceeds VM budgetgit

为了解决这个问题,就出现了Bitmap的高效加载策略。其实核心思想很简单。假设
经过ImageView来显示图片,不少时候ImageView并无原始图片的尺寸那么大,
这个时候把整个图片加载进来后再设置给ImageView,显然是没有必要的,由于
ImageView根本没办法显示原始图片。这时候就能够按必定的采样率来将图片缩小
后再加载进来,这样图片既能在ImageView显示出来,又能下降内存占用从而在一
定程度上避免OOM,提升了Bitmap加载时的性能github

2、Bitmap高效加载的具体方式

1.加载Bitmap的方式

Bitmap在Android中指的是一张图片。经过BitmapFactory类提供的四类方法:
decodeFile,decodeResource,decodeStreamdecodeByteArray,分别从文件系统,
资源,输入流和字节数组中加载出一个Bitmap对象,其中
decodeFile,decodeResource又间接调用了decodeStream方法,这四类方法最终是
在Android的底层实现的,对应着BitmapFactory类的几个native方法。数组

2.BitmapFactory.Options的参数

inSampleSize参数
上述四类方法都支持BitmapFactory.Options参数,而Bitmap的按必定采样率进行缩
放就是经过BitmapFactory.Options参数实现的,主要用到了inSampleSize参数,即
采样率。经过对inSampleSize的设置,对图片的像素的高和宽进行缩放。ide

inSampleSize=1,即采样后的图片大小为图片的原始大小。小于1,也按照1来计
算。 当inSampleSize>1,即采样后的图片将会缩小,缩放比例为1/(inSampleSize
的二次方)。性能

例如: 一张1024 ×1024像素的图片,采用ARGB8888格式存储,那么内存大小
1024×1024×4=4M。若是inSampleSize=2,那么采样后的图片内存大小:
512×512×4=1M。code

注意: 官方文档支出,inSampleSize的取值应该老是2的指数,如1,2,4,8等。
若是外界传入的inSampleSize的值不为2的指数,那么系统会向下取整并选择一个
最接近2的指数来代替。好比3,系统会选择2来代替。当时经验证实并不是在全部
Android版本上都成立。对象

关于inSampleSize取值的注意事项: 一般是根据图片宽高实际的大小/须要的宽高
大小,分别计算出宽和高的缩放比。但应该取其中最小的缩放比,避免缩放图片太
小,到达指定控件中不能铺满,须要拉伸从而致使模糊。图片

例如: ImageView的大小是100×100像素,而图片的原始大小为200×300,那么宽
的缩放比是2,高的缩放比是3。若是最终inSampleSize=2,那么缩放后的图片大小
100×150,仍然合适ImageView。若是inSampleSize=3,那么缩放后的图片大小小
ImageView所指望的大小,这样图片就会被拉伸而致使模糊。ip

inJustDecodeBounds参数
咱们须要获取加载的图片的宽高信息,而后交给inSampleSize参数选择缩放比缩
放。那么如何能先不加载图片却能得到图片的宽高信息,经过
inJustDecodeBounds=true,而后加载图片就能够实现只解析图片的宽高信息,并
不会真正的加载图片,因此这个操做是轻量级的。当获取了宽高信息,计算出缩放
比后,而后在将inJustDecodeBounds=false,再从新加载图片,就能够加载缩放后
的图片。

注意: BitmapFactory获取的图片宽高信息和图片的位置以及程序运行的设备有
关,好比同一张图片放在不一样的drawable目录下或者程序运行在不一样屏幕密度的设
备上,均可能致使BitmapFactory获取到不一样的结果,和Android的资源加载机制
有关

3.高效加载Bitmap的流程

①将BitmapFactory.OptionsinJustDecodeBounds参数设为true并加载图片。
②从BitmapFactory.Options中取出图片的原始宽高信息,它们对应于outWidth
outHeight参数。

③根据采样率的规则并结合目标View的所需大小计算出采样率inSampleSize
④将BitmapFactory.OptionsinJustDecodeBounds参数设为false,而后从新加载
图片。

3、Bitmap高效加载的代码实现

public static Bitmap decodeSampledBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight){
     BitmapFactory.Options options = new BitmapFactory.Options();
     options.inJustDecodeBounds = true;
     //加载图片
     BitmapFactory.decodeResource(res,resId,options);
     //计算缩放比
     options.inSampleSize = calculateInSampleSize(options,reqHeight,reqWidth);
     //从新加载图片
     options.inJustDecodeBounds =false;
     return BitmapFactory.decodeResource(res,resId,options);
  }

  private static int calculateInSampleSize(BitmapFactory.Options options, int reqHeight, int reqWidth) {
     int height = options.outHeight;
     int width = options.outWidth;
     int inSampleSize = 1;
     if(height>reqHeight||width>reqWidth){
         int halfHeight = height/2;
         int halfWidth = width/2;
         //计算缩放比,是2的指数
         while((halfHeight/inSampleSize)>=reqHeight&&(halfWidth/inSampleSize)>=reqWidth){
               inSampleSize*=2;
         }
     }

     return inSampleSize;
  }

这个时候就能够经过以下方式高效加载图片:

mImageView.setImageBitmap(decodeSampledBitmapFromResource(getResources(),R.mipmap.ic_launcher,100,100);

除了BitmapFactorydecodeResource方法,其余方法也能够相似实现。

更多内容详情请关注个人GitHub:https://github.com/xiangjiana/Android-MS

相关文章
相关标签/搜索