【Google官方教程】第一课:高效地加载大Bitmap(位图)
转载声明:Ryan的博客文章欢迎您的转载,但在转载的同时,请注明文章的来源出处,不胜感激! :-)
http://blog.csdn.net/floodingfire/article/details/8168114
译者:Ryan Hoo
来源:https://developer.android.com/develop/index.html
译者按: 在Google最新的文档中,提供了一系列含金量相当高的教程。因为种种原因而鲜为人知,真是可惜!Ryan将会细心整理,将之翻译成中文,希望对开发者有所帮助。
本系列是Google关于展示大Bitmap(位图)的官方演示,可以有效的解决内存限制,更加有效的加载并显示图片,同时避免让人头疼的OOM(Out Of Memory)。
-------------------------------------------------
译文:
图像可以有各种各样的形状和大小。在很多情况下,它们往往会比典型的应用UI界面所需要的更大。例如,系统的Gallery程序展示使用Android设备的摄像头拍摄的照片的分辨率往往要远高于设备的屏幕密度。读取Bitmap(位图)的尺寸和类型
BitmapFactory提供了几种解码方式(decodeByteArray(), decodeFile(), decodeResource()等等),以便从多种资源中创建一个Bitmap(位图)对象。可以根据你的图片数据来源选择最合适的解码方式。这些方法视图为构造Bitmap对象分配内存,因此很容易导致OutOfMemory(OOM)异常。每一种解码方式都有额外的特征,你可以通过BitmapFactory.Options类类指定解码方法。在解码图片的时候设置inJustDecodeBounds属性为true,可以避免内存分配,返回的bitmap对象为null却可以设置outWidth, outHeight和outMimeType。这项技术允许你在创建Bitmap(并分配内存)之前读取图片的尺寸和类型。
1
BitmapFactory.Options options =
new
BitmapFactory.Options();
2
options.inJustDecodeBounds =
true
;
3
BitmapFactory.decodeResource(getResources(), R.id.myimage, options);
4
int
imageHeight = options.outHeight;
5
int
imageWidth = options.outWidth;
6
String imageType = options.outMimeType;
为了避免java.lang.OutOfMemeory异常,在解码图片之前就要检查图片的尺寸,除非你十分确信图片资源的尺寸是可预见的并且有着充裕的可用内存。 例如,如果最终只是要在ImageView中显示一张128*96px大小的缩略图,直接加载1024*768px的图片是非常不值得的。
为了告诉解码器如何对图像进行采样,加载更小版本的图片,需要在BitmapFactory.Options对象中将inSampleSize设置为true。例如,一张分辨率为2048*1536px的图像使用inSampleSize值为4的设置来解码,产生的Bitmap大小约为512*384px。相较于完整图片占用12M的内存,这种方式只需0.75M内存(假设Bitmap配置为ARGB_8888)。这里有一个方法用来计算基于目标高宽的sample size的值:
01
public
static
int
calculateInSampleSize(
02
BitmapFactory.Options options,
int
reqWidth,
int
reqHeight) {
03
// Raw height and width of image
04
final
int
height = options.outHeight;
05
final
int
width = options.outWidth;
06
int
inSampleSize =
1
;
07
08
if
(height > reqHeight || width > reqWidth) {
09
if
(width > height) {
10
inSampleSize = Math.round((
float
)height / (
float
)reqHeight);
11
}
else
{
12
inSampleSize = Math.round((
float
)width / (
float
)reqWidth);
13
}
14
}
15
return
inSampleSize;
16
}
提示:使用2的次幂来设置inSampleSize值可以使解码器执行地更加迅速、更加高效。但是,如果你想在内存或者硬盘上缓存一个调整过大小的图片,通常还是解码到合适的图片尺寸更加节省空间。
要使用这个方法,首先要使用inJustDecodeBounds为true来解码尺寸信息,将options传递过去使用新的inSampleSize值再次解码并且要将inJustDecodeBounds值设置为false。
01
public
static
Bitmap decodeSampledBitmapFromResource(Resources res,
int
resId,
02
int
reqWidth,
int
reqHeight) {
03
04
// First decode with inJustDecodeBounds=true to check dimensions
05
final
BitmapFactory.Options options =
new
BitmapFactory.Options();
06
options.inJustDecodeBounds =
true
;
07
BitmapFactory.decodeResource(res, resId, options);
08
09
// Calculate inSampleSize
10
options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);
11
12
// Decode bitmap with inSampleSize set
13
options.inJustDecodeBounds =
false
;
14
return
BitmapFactory.decodeResource(res, resId, options);
15
}
这个方法使得加载任意大小的Bitmap到展示100*100px缩略图的ImageView中更加简单,如下代码所示: 1
mImageView.setImageBitmap(
2
decodeSampledBitmapFromResource(getResources(), R.id.myimage,
100
,
100
));
你可以根据需要,按照类似的解码过程,采用适当的BitmapFactory.decode*方法从其他资源中解码Bitmap。