高效加载单个大图片——OOM

加载图片时,应用程序占用了过高的内存就容易出现OOM(OutOfMemory)异常

我们可以通过下面的代码看出每个应用程序最高可用内存是多少? 如下:

 int maxMemory = (int)(Runtime.getRuntime().maxMemory() / 1024);  
  Log.d("TAG", "Max memory is " + maxMemory + "KB");  

BitmapFactory这个类提供了多个解析方法(decodeByteArray, decodeFile, decodeResource等)用于创建Bitmap对象,我们应该根据图片的来源选择合适的方法。比如SD卡中的图片可以使用decodeFile方法,网络上的图片可以使用decodeStream方法,资源文件中的图片可以使用decodeResource方法。这些方法会尝试为已经构建的bitmap分配内存,这时就会很容易导致OOM出现。为此每一种解析方法都提供了一个可选的BitmapFactory.Options参数,将这个参数的inJustDecodeBounds属性设置为true就可以让解析方法禁止为bitmap分配内存,返回值也不再是一个Bitmap对象,而是null。虽然Bitmap是null了,但是BitmapFactory.Options的outWidth、outHeight和outMimeType属性都会被赋值。这个技巧让我们可以在加载图片之前就获取到图片的长宽值和MIME类型,从而根据情况对图片进行压缩。如下代码所示:


BitmapFactory.Options options = new BitmapFactory.Options();  
options.inJustDecodeBounds = true;  
BitmapFactory.decodeResource(getResources(), R.id.myimage, options);  
int imageHeight = options.outHeight;  
int imageWidth = options.outWidth;  
String imageType = options.outMimeType; 

比如,你的ImageView只有128*96像素的大小,只是为了显示一张缩略图,这时候把一张1024*768像素的图片完全加载到内存中显然是不值得的。
那我们怎样才能对图片进行压缩呢?通过设置BitmapFactory.Options中inSampleSize的值就可以实现。比如我们有一张2048*1536像素的图片,将inSampleSize的值设置为4,就可以把这张图片压缩成512*384像素。原本加载这张图片需要占用13M的内存,压缩后就只需要占用0.75M了(假设图片是ARGB_8888类型,即每个像素点占用4个字节)。下面的方法可以根据传入的宽和高,计算出合适的inSampleSize值:


 public static int calculateInSampleSize(BitmapFactory.Options options,  
       int reqWidth, int reqHeight) {  
   // 源图片的高度和宽度  
    final int height = options.outHeight;  
   final int width = options.outWidth;  
  int inSampleSize = 1;  
  if (height > reqHeight || width > reqWidth) {  
       // 计算出实际宽高和目标宽高的比率  
      final int heightRatio = Math.round((float) height / (float) reqHeight);  
      final int widthRatio = Math.round((float) width / (float) reqWidth);  
      // 选择宽和高中最小的比率作为inSampleSize的值,这样可以保证最终图片的宽和高 
    // 一定都会大于等于目标的宽和高。  
       inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;  
  }  
  return inSampleSize;  
}   

使用这个方法,首先你要将BitmapFactory.Options的inJustDecodeBounds属性设置为true,解析一次图片。然后将BitmapFactory.Options连同期望的宽度和高度一起传递到到calculateInSampleSize方法中,就可以得到合适的inSampleSize值了。之后再解析一次图片,使用新获取到的inSampleSize值,并把inJustDecodeBounds设置为false,就可以得到压缩后的图片了。


 public static Bitmap decodeSampledBitmapFromResource(Resources res, int resId,  
int reqWidth, int reqHeight) {  
// 第一次解析将inJustDecodeBounds设置为true,来获取图片大小  
final BitmapFactory.Options options = new BitmapFactory.Options();  
 options.inJustDecodeBounds = true;  
  BitmapFactory.decodeResource(res, resId, options);  
  // 调用上面定义的方法计算inSampleSize值  
  options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);  
  // 使用获取到的inSampleSize值再次解析图片  
   options.inJustDecodeBounds = false;  
   return BitmapFactory.decodeResource(res, resId, options);  
}  

也可以将任意一张图片压缩成100*100的缩略图,并在ImageView上展示,如下:


 mImageView.setImageBitmap(   decodeSampledBitmapFromResource(getResources(), R.id.myimage, 100, 100));  

你可能感兴趣的:(Android程序代码)