更高效的加载位图
图像能够体现各种的形状和大小。在许多情况下,它们是都会大于一个典型要求的应用程序的用户界面(UI)。例如,显示照片的系统应用程序Gallery使用了你的Android设备的相机,而它通常比屏幕分辨率比你的设备的屏幕密度更高。
鉴于你的内存有限,我们可以比较理想的只让内存加载较低分辨率的版本。低分辨率版本应与显示它的UI组件的大小相符合。一个高分辨率的图像不提供任何可显示的适配,但仍依然会占用更多的珍贵的内存资源,会带来额外的性能开销上的剧烈增加。
这节课带你浏览解码大型位图并且可以不超过每个应用程序的内存限制而通过在内存中使用一个较小的采样版本加载。
读取位图的尺寸和类型
BitmapFactory
类提供了这几种解码方法:
decodeByteArray()
,
decodeFile()
,
decodeResource()
, 等等.从不同的来源创建
Bitmap
位图对象, 要选择最适合于你的图像数据源的解码方法。 因为可以很容易地触发
OutOfMemory
异常. 每种类型的解码方法具有额外的标志来使用特定
BitmapFactory.Options
解码方式。在解码时通过设置
inJustDecodeBounds
属性为
true来
防止内存分配。当位图对象没有设置
outWidth
,
outHeight
和
outMimeType的时候,会返回null
。这种技巧能够让你读取尺寸和位图在构建前的图像数据(和内存分配)。
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;
为防止java.lang.OutOfMemory
这种异常, 要在对位图进行解码以前检查尺寸, 除非你绝对相信来源提供给你的可以预测的尺寸的图像数据并能够很合适存放在可用的内存中.
加载一张缩小版本的位图到内存中
既然现在图像的尺寸是已知的,它们可以被用来决定是否完整的图像应该被加载到内存或者改为加载二次采样版本。这里有一些需要考虑的因素:
- 加载整张图片到内存中的用量估计.
- 你愿意加载此图像的内存用量给您的应用程序的任何其他的内存需求量。
ImageView
的目标尺寸或者图像要加载到的UI组件.- 当前设备的屏幕大小和密度。
ImageView
中.
要告知解码器重采样的图像, 加载一张更小的版本到内存中, 要设置inSampleSize
为true
在你的BitmapFactory.Options
对象中。例如,一张分辨率为2048x1536用inSampleSize
为4来解码缩减位图大小到 512x384. 加载这张图片使用的内存量只需要 0.75MB 要比加载整张图片使用 12MB内存要好(假设位图配置为ARGB_8888
)。这有一种基于目标宽度和高度来计算样本大小值的方法:
public static int calculateInSampleSize(
BitmapFactory.Options options, int reqWidth, int reqHeight) {
// Raw height and width of image
final int height = options.outHeight;
final int width = options.outWidth;
int inSampleSize = 1;
if (height > reqHeight || width > reqWidth) {
// Calculate ratios 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);
// Choose the smallest ratio as inSampleSize value, this will guarantee
// a final image with both dimensions larger than or equal to the
// requested height and width.
inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio;
}
return inSampleSize;
}
Note:使用2为
inSampleSize
赋值会让解码器更快更有效率. 然而, 如果你计划缓存一个重新调整大小的版本在内存或者磁盘中, 这通常更值得解码到合适的图像大小来节省空间。
要使用这种方法, 第一次解码要使inJustDecodeBounds
值为true
, 传入选项后再一次解码时使用新的inSampleSize
值并将inJustDecodeBounds
设置为false
:
public static Bitmap decodeSampledBitmapFromResource(Resources res, int resId,
int reqWidth, int reqHeight) {
// First decode with inJustDecodeBounds=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 with inSampleSize set
options.inJustDecodeBounds = false;
return BitmapFactory.decodeResource(res, resId, options);
}
这个方法可以很容易的将一张大尺寸图像加载到
ImageView
来显示为100x100像素的缩略图, 如下为示例代码:
mImageView.setImageBitmap(
decodeSampledBitmapFromResource(getResources(), R.id.myimage, 100, 100));
你可以按照类似过程,解码其他来源的位图,通过使用
BitmapFactory.decode*
一类的方法来实现需求.
源地址:http://developer.android.com/intl/zh-CN/training/displaying-bitmaps/load-bitmap.html