java.lang.OutOfMemoryError: bitmap size exceeds VM budget解決方法

在開發圖片瀏覽器等軟件是,很多時候要顯示圖片的縮略圖,而一般情況下,我們要將圖片按照固定大小取縮略圖,一般取縮略圖的方法是使用BitmapFactory的decodeFile方法,然後通過傳遞進去BitmapFactory.Option類型的參數進行取縮略圖,在Option中,屬性值inSampleSize表示縮略圖大小爲原始圖片大小的幾分之一,即如果這個值爲2,則取出的縮略圖的寬和高都是原始圖片的1/2,圖片大小就爲原始大小的1/4。

  然而,如果我們想取固定大小的縮略圖就比較困難了,比如,我們想將不同大小的圖片去出來的縮略圖高度都爲200px,而且要保證圖片不失真,那怎麼辦?我們總不能將原始圖片加載到內存中再進行縮放處理吧,要知道在移動開發中,內存是相當寶貴的,而且一張100K的圖片,加載完所佔用的內存何止100K?

  經過研究,發現,Options中有個屬性inJustDecodeBounds,研究了一下,終於明白是什麼意思了,SDK中的E文是這麼說的

  If set to true, the decoder willreturn null (no bitmap), but the out... fields will still be set,allowing the caller to query the bitmap without having to allocatethe memory for its pixels.

  意思就是說如果該值設爲true那麼將不返回實際的bitmap不給其分配內存空間而裏面只包括一些解碼邊界信息即圖片大小信息,那麼相應的方法也就出來了,通過設置inJustDecodeBounds爲true,獲取到outHeight(圖片原始高度)和outWidth(圖片的原始寬度),然後計算一個inSampleSize(縮放值),然後就可以取圖片了,這裏要注意的是,inSampleSize可能小於0,必須做判斷。
具體代碼如下:
FrameLayoutfr=(FrameLayout)findViewById(R.id.FrameLayout01);
      BitmapFactory.Options options = newBitmapFactory.Options();
      options.inJustDecodeBounds = true;
       //獲取這個圖片的寬和高
       Bitmapbitmap = BitmapFactory.decodeFile("/sdcard/test.jpg", options);//此時返回bm爲空
      options.inJustDecodeBounds = false;
        //計算縮放比
       int be =(int)(options.outHeight / (float)200);
       if (be<= 0)
          be =1;
      options.inSampleSize = be;
       //重新讀入圖片,注意這次要把options.inJustDecodeBounds 設爲false哦
      bitmap=BitmapFactory.decodeFile("/sdcard/test.jpg",options);
       int w =bitmap.getWidth();
       int h =bitmap.getHeight();
      System.out.println(w+"  "+h);
       ImageViewiv=new ImageView(this);
      iv.setImageBitmap(bitmap);
這樣我們就可以讀取較大的圖片而不會內存溢出了。如果你想把壓縮後的圖片保存在Sdcard上的話就很簡單了:
File file=newFile("/sdcard/feng.png");
       try{
         FileOutputStream out=new FileOutputStream(file);
         if(bitmap.compress(Bitmap.CompressFormat.PNG, 100,out)){
            out.flush();
            out.close();
         }
       } catch(FileNotFoundException e) {
          // TODOAuto-generated catch block
         e.printStackTrace();
       } catch(IOException e) {
          // TODOAuto-generated catch block
         e.printStackTrace();
      }


ok,這樣就把圖片保存在/sdcard/feng.png這個文件裏面了,呵呵。
***********************************************************************************

BitmapFactory.decodeFile(imageFile);

用BitmapFactory解碼一張圖片時,有時會遇到該錯誤。這往往是由於圖片過大造成的。要想正常使用,則需要分配更少的內存空間來存儲。

BitmapFactory.Options.inSampleSize

BitmapFactory.Options opts = new BitmapFactory.Options();

opts.inSampleSize = 4;

Bitmap bitmap = BitmapFactory.decodeFile(imageFile, opts);

BitmapFactory.Options opts = new BitmapFactory.Options();

opts.inSampleSize = 4;

Bitmap bitmap = BitmapFactory.decodeFile(imageFile, opts);


設置恰當的inSampleSize可以使BitmapFactory分配更少的空間以消除該錯誤。inSampleSize的具體含義請參考SDK文檔。例如:設置恰當的inSampleSize是解決該問題的關鍵之一。BitmapFactory.Options提供了另一個成員inJustDecodeBounds。

BitmapFactory.Options opts = new BitmapFactory.Options();

opts.inJustDecodeBounds = true;

Bitmap bitmap = BitmapFactory.decodeFile(imageFile, opts);

設置inJustDecodeBounds爲true後,decodeFile並不分配空間,但可計算出原始圖片的長度和寬度,即opts.width和opts.height。有了這兩個參數,再通過一定的算法,即可得到一個恰當的inSampleSize。

查看Android源碼,Android提供了一種動態計算的方法。

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;

    }

} 

使用該算法,就可動態計算出圖片的inSampleSize。

BitmapFactory.Options opts = new BitmapFactory.Options();

opts.inJustDecodeBounds = true;

BitmapFactory.decodeFile(imageFile, opts);

   

opts.inSampleSize = computeSampleSize(opts, -1, 128*128);  

opts.inJustDecodeBounds = false;

try {

 Bitmap bmp = BitmapFactory.decodeFile(imageFile, opts);

 imageView.setImageBitmap(bmp);

    } catch (OutOfMemoryError err) {

    }

另外,可以通過Bitmap.recycle()方法來釋放位圖所佔的空間,當然前提是位圖沒有被使用。

轉載自:http://blog.sina.com.cn/s/blog_672f41790100t35n.html

發佈了22 篇原創文章 · 獲贊 3 · 訪問量 4萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章