android BitmapFactory.Options

本文介绍如何使用BitmapFactory.Options类获取图片尺寸并避免内存溢出。通过设置inJustDecodeBounds为true,可以只获取图片的宽高信息而不加载整个图片,从而减少内存占用。此外,还介绍了如何根据屏幕尺寸调整图片大小。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

下面我们回到我们的主题上来:怎样获取图片的大小?
思路很简单:
首先我们把这个图片转成Bitmap,然后再利用Bitmap的getWidth()和getHeight()方法就可以取到图片的宽高了。
新问题又来了,在通过BitmapFactory.decodeFile(Stringpath)方法将突破转成Bitmap时,遇到大一些的图片,我们经常会遇到OOM(Out OfMemory)的问题。怎么避免它呢?
这就用到了我们上面提到的BitmapFactory.Options这个类。
 
BitmapFactory.Options这个类,有一个字段叫做 inJustDecodeBounds 。SDK中对这个成员的说明是这样的:
If set to true, the decoderwill return null (no bitmap), but theout…
也就是说,如果我们把它设为true,那么BitmapFactory.decodeFile(Stringpath, Optionsopt)并不会真的返回一个Bitmap给你,它仅仅会把它的宽,高取回来给你,这样就不会占用太多的内存,也就不会那么频繁的发生OOM了。
示例代码如下:
 
BitmapFactory.Options options = newBitmapFactory.Options();
options.inJustDecodeBounds = true;
Bitmap bmp = BitmapFactory.decodeFile(path, options);
复制代码
 
这段代码之后,options.outWidth 和 options.outHeight就是我们想要的宽和高了。
 
有了宽,高的信息,我们怎样在图片不变形的情况下获取到图片指定大小的缩略图呢?
比如我们需要在图片不变形的前提下得到宽度为200的缩略图。
那么我们需要先计算一下缩放之后,图片的高度是多少 
 
int height = options.outHeight * 200 / options.outWidth;
options.outWidth = 200;
options.outHeight = height; 
options.inJustDecodeBounds = false;
Bitmap bmp = BitmapFactory.decodeFile(path, options);
image.setImageBitmap(bmp);
复制代码
 
这样虽然我们可以得到我们期望大小的ImageView
但是在执行BitmapFactory.decodeFile(path,options);时,并没有节约内存。要想节约内存,还需要用到BitmapFactory.Options这个类里的 inSampleSize 这个成员变量。
我们可以根据图片实际的宽高和我们期望的宽高来计算得到这个值。
 
inSampleSize = options.outWidth / 200;
另外,为了节约内存我们还可以使用下面的几个字段:
 
options.inPreferredConfig =Bitmap.Config.ARGB_4444;    //默认是Bitmap.Config.ARGB_8888
options.inPurgeable = true;
options.inInputShareable = true;
 
加载和显示图片是很消耗内存的一件事,BitmapFactory.Options 类,  允许我们定义图片以何种方式如何读到内存,
 
[html
BitmapFactory.Options bmpFactoryOptions = new BitmapFactory.Options();    
bmpFactoryOptions.inSampleSize = 8;    
Bitmap bmp = BitmapFactory.decodeFile(imageFilePath, bmpFactoryOptions);    
imv.setImageBitmap(bmp);    
上面的代码使图片变成原来的1/8.
[html] 
//imv = (ImageView) findViewById(R.id.ReturnedImageView);    
        Display currentDisplay = getWindowManager().getDefaultDisplay();    
        int dw = currentDisplay.getWidth();    
        int dh = currentDisplay.getHeight();    
          try    
       {    
        BitmapFactory.Options bmpFactoryOptions = new BitmapFactory.Options();    
        bmpFactoryOptions.inJustDecodeBounds = true;    
        Bitmap bmp = BitmapFactory.decodeStream(getContentResolver().    
                openInputStream(imageFileUri), null,  bmpFactoryOptions);    
    
        int heightRatio = (int)Math.ceil(bmpFactoryOptions.outHeight/(float)dh);    
        int widthRatio = (int)Math.ceil(bmpFactoryOptions.outWidth/(float)dw);    
    
        Log.v("HEIGHRATIO", ""+heightRatio);    
        Log.v("WIDTHRATIO", ""+widthRatio);    
    
        if (heightRatio > 1 && widthRatio > 1)    
        {    
            bmpFactoryOptions.inSampleSize =  heightRatio > widthRatio ? heightRatio:widthRatio;    
        }    
        bmpFactoryOptions.inJustDecodeBounds = false;    
        bmp = BitmapFactory.decodeStream(getContentResolver().    
                openInputStream(imageFileUri), null,  bmpFactoryOptions);    
           returnedImageView.setImageBitmap(bmp);    
       }    
       catch (FileNotFoundException e)    
       {    
           Log.v("ERROR", e.toString());    
    
       }    
上面的代码让图片根据窗口大小改变
bmpFactoryOptions.inJustDecodeBounds = true; 这一行让代码只解码图片的Bounds
 
如果设置为真,译码器将返回null(没有图),但是……领域仍将被设置,允许调用者来查询图而无需为其像素分配内存。
 
1.inSampleSize(设置缩放为原来的四分之一)
[html]  
BitmapFactory.Options opts = new BitmapFactory.Options();  
opts.inSampleSize = <span style="background-color:rgb(255,204,153)">4</span>;  
Bitmap bitmap = BitmapFactory.decodeFile(imageFile, opts);  
2.inJustDecodeBounds
[html]  
BitmapFactory.Options opts = new BitmapFactory.Options();  
opts.inJustDecodeBounds = true;  
Bitmap bitmap = BitmapFactory.decodeFile(imageFile, opts);  
设置inJustDecodeBounds为true后,decodeFile并不分配空间,但可计算出原始图片的长度和宽度,即opts.width和opts.height。有了这两个参数,再通过一定的算法,即可得到一个恰当的inSampleSize。
综合上述(1,2),完整代码是
[html]  
public static Bitmap createImageThumbnail(String filePath){    
     Bitmap bitmap = null;    
     BitmapFactory.Options opts = new BitmapFactory.Options();    
     opts.inJustDecodeBounds = true;    
     BitmapFactory.decodeFile(filePath, opts);    
    
     opts.inSampleSize = computeSampleSize(opts, -1, 128*128);    
     opts.inJustDecodeBounds = false;    
    
     try {    
         bitmap = BitmapFactory.decodeFile(filePath, opts);    
     }catch (Exception e) {    
        // TODO: handle exception    
    }    
    return bitmap;    
}    
    
public static int computeSampleSize(BitmapFactory.Options options, int minSideLength, int maxNumOfPixels) {    
    int initialSize = computeInitialSampleSize(options, minSideLength, maxNumOfPixels);    
    int roundedSize;    
    if (initialSize <= 8) {    
        roundedSize = 1;    
        while (roundedSize < initialSize) {    
            roundedSize <<= 1;    
        }    
    } else {    
        roundedSize = (initialSize + 7) / 8 * 8;    
    }    
    return roundedSize;    
}    
    
private static int computeInitialSampleSize(BitmapFactory.Options options,int minSideLength, int maxNumOfPixels) {    
    double w = options.outWidth;    
    double h = options.outHeight;    
    int lowerBound = (maxNumOfPixels == -1) ? 1 : (int) Math.ceil(Math.sqrt(w * h / maxNumOfPixels));    
    int upperBound = (minSideLength == -1) ? 128 :(int) Math.min(Math.floor(w / minSideLength), Math.floor(h / minSideLength));    
    if (upperBound < lowerBound) {    
        // return the larger one when there is no overlapping zone.    
        return lowerBound;    
    }    
    if ((maxNumOfPixels == -1) && (minSideLength == -1)) {    
        return 1;    
    } else if (minSideLength == -1) {    
        return lowerBound;    
    } else {    
        return upperBound;    
    }    
}   
————————————————————————————————
    BitmapFactory.Options这个类的信息:http://developer.android.com/reference/android/graphics/BitmapFactory.Options.html

其相关信息如下:


这个表格是从android sdk文档里摘出来的,简单看一下说明就明白是什么意思了。
下面我们回到我们的主题上来:怎样获取图片的大小?
思路很简单:
首先我们把这个图片转成Bitmap,然后再利用Bitmap的getWidth()和getHeight()方法就可以取到图片的宽高了。
新问题又来了,在通过BitmapFactory.decodeFile(String path)方法将突破转成Bitmap时,遇到大一些的图片,我们经常会遇到OOM(Out Of Memory)的问题。怎么避免它呢?
这就用到了我们上面提到的BitmapFactory.Options这个类。
BitmapFactory.Options这个类,有一个字段叫做 inJustDecodeBounds 。SDK中对这个成员的说明是这样的:
If set to true, the decoder will return null (no bitmap), but the out…
也就是说,如果我们把它设为true,那么BitmapFactory.decodeFile(String path, Options opt)并不会真的返回一个Bitmap给你,它仅仅会把它的宽,高取回来给你,这样就不会占用太多的内存,也就不会那么频繁的发生OOM了。
示例代码如下:

  1. BitmapFactory.Options options = new BitmapFactory.Options();  
  2. options.inJustDecodeBounds = true;  
  3. Bitmap bmp = BitmapFactory.decodeFile(path, options);  
  4. /* 这里返回的bmp是null */  
这段代码之后,options.outWidth 和 options.outHeight就是我们想要的宽和高了。
有了宽,高的信息,我们怎样在图片不变形的情况下获取到图片指定大小的缩略图呢?
比如我们需要在图片不变形的前提下得到宽度为200的缩略图。
那么我们需要先计算一下缩放之后,图片的高度是多少 
  1. /* 计算得到图片的高度 */  
  2. /* 这里需要主意,如果你需要更高的精度来保证图片不变形的话,需要自己进行一下数学运算 */  
  3. int height = options.outHeight * 200 / options.outWidth;  
  4. options.outWidth = 200;  
  5. options.outHeight = height;   
  6. /* 这样才能真正的返回一个Bitmap给你 */  
  7. options.inJustDecodeBounds = false;  
  8. Bitmap bmp = BitmapFactory.decodeFile(path, options);  
  9. image.setImageBitmap(bmp);  
这样虽然我们可以得到我们期望大小的ImageView
但是在执行BitmapFactory.decodeFile(path, options);时,并没有节约内存。要想节约内存,还需要用到BitmapFactory.Options这个类里的 inSampleSize 这个成员变量。
我们可以根据图片实际的宽高和我们期望的宽高来计算得到这个值。
  1. inSampleSize = options.outWidth / 200;  
另外,为了节约内存我们还可以使用下面的几个字段:
  1. options.inPreferredConfig = Bitmap.Config.ARGB_4444;    // 默认是Bitmap.Config.ARGB_8888  
  2. /* 下面两个字段需要组合使用 */  
  3. options.inPurgeable = true;  
  4. options.inInputShareable = true;  

参考资料:

BitmapFactory.Options详解


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值