正文  高级教程 > 程序优化 >

Android解决大图片内存溢出

在Android平台上内存溢出是一件很头疼的事情,但解决办法还是有的,今天来翻译一篇谷歌官方提供的解决方案,原文可以参见:http://developer.android.com/training/displaying-bitmaps/load-bitmap.html 。更有效......

在Android平台上内存溢出是一件很头疼的事情,但解决办法还是有的,今天来翻译一篇谷歌官方提供的解决方案,原文可以参见:http://developer.android.com/training/displaying-bitmaps/load-bitmap.html 。

 更有效地加载大的图片

图片有不同的形状可大小,在很多情况下他们总是大于一个典型应用UI的要求。例如,系统提供的Gallery程序显示了手机相机拍摄的图片,这些图片的分辨率总是大于设备屏幕的分辨率。

考虑到跟有限的设备内存打交道,理想情况下,你只需要加载在内存中的较低分辨率版本。分辨率较低的UI组键需要与显示它的大小相匹配。具有更高的分辨率的图像并不会带来显而易见的好处,但是仍然占用了宝贵的内存和因为缩放带来了额外的性能开销。

这节课将要通过载入一个较小的二次取样的小图片带你来编码一个大的图片而不需要超过每个程序最大的内存。

 读取图片的尺寸和类型

BitmapFactory类提供了一些列创建不同图片种类的编码方式,例如:decodeByteArray(), decodeFile(), decodeResource() 等 等。基于你的图片数据来源选择一个最合适的编码方式。这些编码方法试图为创建Bitmap分配内存,因此可以很容易地导致OutOfMemory异常。每 个类型的解码方法有额外的签名,让您指定通过BitmapFactory.Options类的解码选项。通过设置 BitmapFactory.Options的inJustDecodeBounds属性设置为true,可以解码避免内存分配,返回的Bitmap为 空,但返回outWidth,outHeight和outMimeType。这种技术使您可以读取的图像数据的尺寸和类型在内存分配之前。

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.OutOfMemory异常,检查解码前的位图的尺寸,除非您绝对信任的来源,以提供可预见的大小的图像数据,适合有限的内存空间。

 加载到内存中的一个缩减版本

现在的图像尺寸是已知的,它们可以被用来决定是否应加载到内存或二次采样的版本,而不是应装入完整的图像。这里有一些要考虑的因素:

  • 估计载入一张完整图像的内存使用

  • 你愿意投入到这个图片加载应用程序的任何其他内存要求的内存量。

  • 图像要加载到的目标的ImageView或UI组件的尺寸。

  • 当前设备的屏幕大小和密度

例如,没有必要把一个1024×768像素的图像加载到内存中,如果它最终会被显示在一个ImageView 128×96像素的缩略图。告诉图像解码器,加载到内存中的一个较小的版本,设置BitmapFactory.Options对象中的 inSampleSize。例如,与2048×1536分辨率的图像,设置其inSampleSize为4(即Width为1/4,Height为1 /4,整个图像大小为1/16)将产生一个约512×384的位图。载入到内存使用完整的图像(假设ARGB_8888位图配置)0.75MB,而超过 12MB。这里有一个方法来计算的样本大小目标的宽度和高度上的值:

01

public static int calculateInSampleSize(

02

            BitmapFactory.Options options, int reqWidth, int reqHeight) {

03

    // 图像原始高度和宽度

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

}

注意: inSampleSize 设置为2的平方则编码速度更快更高效。不过,如果你打算在内存或磁盘上的缓存调整过大小的版本,它通常还是值得解码到最合适的图像尺寸,以节省空间。

设置为true inJustDecodeBounds使用这种方法,首先解码,通过选项,然后解码再次使用新inSampleSize值并inJustDecodeBounds设置为false:

01

public static Bitmap decodeSampledBitmapFromResource(Resources res, int resId,

02

        int reqWidth, int reqHeight) {

03

 

04

    // 首先设置 inJustDecodeBounds=true 来检查尺寸

05

    final BitmapFactory.Options options = new BitmapFactory.Options();

06

    options.inJustDecodeBounds = true;

07

    BitmapFactory.decodeResource(res, resId, options);

08

 

09

    // 计算压缩比例

10

    options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight);

11

 

12

    // 设置inJustDecodeBounds为false

13

    options.inJustDecodeBounds = false;

14

    return BitmapFactory.decodeResource(res, resId, options);

15

}

这种方法使很容易地装入一个ImageView显示一个100×100像素的缩略图任意大尺寸位图,在下面的示例代码所示:

1

mImageView.setImageBitmap(

2

 

3

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

,你可以按照一个类似的过程,使用不同的BitmapFactory.decode*方法来编码其他来源的图像。