图片有各种形状和尺寸。在很多情况下,它们通常要大于显示图片用的用户界面(UI)。例如,系统的Gallery应用程序会显示Android设备的照相机所拍摄的照片,通常照片的分辨率要高于设备的屏幕密度。
由于内存的限制,你可能只想在内存中加载较低分辨率的图片版本。低分辨率版本应该跟显示它的UI组件尺寸相匹配。高分辨率的图片不会有益于显示,但是依然会消耗高贵的内存,并且由于缩放还会导致性能下降。
本文讲解如何用较小的采样版本来解码大位图,而不让其超过每个应用程序内存的限制。
读取位图的尺寸和类型
BitmapFactory类提供了几个解码方法(如decodeByteArray()、decodeFile()、decodeResource()等)来创建来自各种资源的位图(Bitmap)。基于图片数据源来选择最恰当的解码方法。这些方法尝试着给被创建的位图分配内存,内存很容易产生OutOfMemory异常。每种类型的解码方法都有一个附加的标识,这个标识会通过BitmapFactory.Options类让你指定解码选项。
把inJustDecodeBounds属性设置为true,解码时会避免过度的内存分配,除了设置outWidth、outHeight和outMimeType属性设置以外的位图对象都会返回null。这种技术会允许你在构造(和分配内存)位图之前读取图片数据打的尺寸和类型。
BitmapFactory.Options options =newBitmapFactory.Options();
options.inJustDecodeBounds= true;
BitmapFactory.decodeResource(getResources(), R.id.myimage, options);
int imageHeight = options.outHeight;
int imageWidth = options.outWidth;
String imageType = options.outMimeType;
要避免java.lang.OutOfMemory异常,就要在解码之前检查位图的尺寸,除非你确信提供给你的图片数据源能够适合可用的内存。
把缩小的图片加载到内存中
获取图片的尺寸之后,就可以用它来判断是否可以把图片完整的加载到内存中,或者是降低采样率来加载图片。以下是要考虑的一些因素:
1. 估算把图片完整的加载到内存时所有使用的内存;
2. 除了图片的内存需求之外,应用程序的其他内存需求;
3. 显示图片的UI组件的尺寸;
4. 当前设备的屏幕尺寸和密度。
例如,把一个1024x768像素的图片完全加载到内存中,并在一个128x96像素的ImageView组件中,这种内存消耗是不值得的。
设置BitmapFactory.Options对象的inSampleSize属性,就可以告诉解码器降低采样率,把较小的分辨率的图片加载到内存中。例如,inSampleSize属性设置为4,那么分辨率为2048x1536的图片就可以解码为512x384像素的位图。这时所消耗的内存是0.75MB而不是12MB(假设位图是ARGB_8888形态)。以下是基于目标的宽度和高度来计算采样尺寸值的方法:
publicstaticintcalculateInSampleSize(
BitmapFactory.Options options, int reqWidth, int reqHeight) {
// Raw height and width ofimage
final int height = options.outHeight;
final int width = options.outWidth;
intinSampleSize = 1;
if (height > reqHeight || width > reqWidth) {
// Calculateratios of height and width to requested height and width
final int heightRatio = Math.round((float) height / (float) reqHeight);
final int widthRatio = Math.round((float) width / (float) reqWidth);
// Choosethe smallest ratio as inSampleSize value, this will guarantee
// a finalimage with both dimensions larger than or equal to the
// requestedheight and width.
inSampleSize = heightRatio < widthRatio ?heightRatio : widthRatio;
}
return inSampleSize;
}
注意:inSampleSize值使用2的倍数会使解码器更快、更高效。但是,如果你只是打算在内存或硬盘上缓存调整后版本,那么通常采用最恰当的图片尺寸来解码,以便节省存储空间,这依然是值得的。
要使用这种方法,首先要把inJustDecodeBounds设置为true进行一次解码,然后再使用新的inSampleSize的值来再次解码,并把inJustDecodeBounds设置为false:
publicstaticBitmapdecodeSampledBitmapFromResource(Resources res,int resId,
int reqWidth, int reqHeight) {
// First decode withinJustDecodeBounds=true to check dimensions
final BitmapFactory.Options options = new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeResource(res, resId, options);
// Calculate inSampleSize
options.inSampleSize =calculateInSampleSize(options, reqWidth, reqHeight);
// Decode bitmap withinSampleSize set
options.inJustDecodeBounds = false;
return BitmapFactory.decodeResource(res, resId, options);
这种方法使得把任意大尺寸的图片加载到一个显示100x100像素的小ImageView中来显示变得很容易。例如:
mImageView.setImageBitmap(
decodeSampledBitmapFromResource(getResources(), R.id.myimage, 100, 100));
你可以使用类似过程对来自其他类型资源的位图进行解码,只需用相应的BitmapFactory.decode*方法来代替就可以了。