译文出自谷歌安卓官方:https://developer.android.com/training/displaying-bitmaps/load-bitmap.html#read-bitmap
图片在大小与形状上各种各样。在许多情况下,它们比应用程序中的UI所需要的尺寸都大。例如,手机自带的相册会为我们展示
相机所拍摄的图片,但这些图片比设备屏幕分辨率高的多。
考虑应用程序内存的限制,理想情况下你只想将低分辨率的图片加载到内存中。低分辨率的图片应要求能够与你所展示的UI大小相同。
高分辨率的图片不仅无法为你带来视觉上的好处,而且还会占用你宝贵的内存并引发额外的内存开销。
本课程将引导你通过加载更小的下采样的图片来解码大的图片,这样可以防止超出应用程序的内存限制。
读取位图的尺寸与类型
BitmapFactory为创造来自不同来源的Bitmap提供了一些解码方法,如decodeByteArray(),decodeFile(),decodeResource()等。你要通过你的图片来源选择合适的解码方法。这些解码方法会为你的位图分配内存空间并且很容易导致内存不足异常(OutOfMemory exception)。
每个解码方法具有一些选项属性,你可以通过BitmapFactory.Option类来指定解码选项。设置选项中的inJustDecodeBounds属性,你可以在
解码的过程中无需为位图分配内存空间,其返回的是空位图对象并设置了outWidth,outHeight,outMineType三个选项的值。这个技巧允许你在构造位图之前,可以先读取图片的大小与图片数据的类型。
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;
这主要是为了避免内存不足异常,所以在解码图片之前需要先检查其大小,除非你完全相信所获取的图片的尺寸是固定的并且不会造成内存异常现象的发生。
加载缩放的图片进入内存中
现在图片的尺寸我们已经知道了,这样我们可以决定是否将整个图片加载到内存中或者是否缩放后再加载。以下是要考虑的因素:
- 估计加载整个图片到内存中的内存使用情况
- 考虑你应用程序的其它内存需求,你愿意为你的图片花费多大的内存
- 考虑展示图片的控件的尺寸
- 考虑当前设备的屏幕分辨率
例如:不值得将1024*768大小的图片加载到最终显示128x96大小缩略图的ImageView中
因而,我们必须告诉解码器去下采样图片(其实就是缩小图片),然后将缩小后的图片加载进入内存中。这需要对BitmapFactory.Options的inSampleSize进行设置,如 2048x1536大小的图片通过设置inSampleSize大小为4,缩放成 512x384大小,这样实际消耗内存仅为0.75M(不缩放为12M,这里图片格式假设为ARGB_8888)。可以根据目标图片的大小来计算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 halfHeight = height / 2;
final int halfWidth = width / 2;
while ((halfHeight / inSampleSize) > reqHeight
&& (halfWidth / inSampleSize) > reqWidth) {
inSampleSize *= 2;
}
}
return inSampleSize;
}
public static Bitmap decodeSampledBitmapFromResource(Resources res, int resId,
int reqWidth, int reqHeight) {
final BitmapFactory.Options optiOns= new BitmapFactory.Options();
options.inJustDecodeBounds = true;
BitmapFactory.decodeResource(res, resId, options);
options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
options.inJustDecodeBounds = false;
return BitmapFactory.decodeResource(res, resId, options);
}
mImageView.setImageBitmap(decodeSampledBitmapFromResource(getResources(), R.id.myimage, 100, 100))