Android之高效載入大圖
載入大圖到記憶體是一件令人頭疼的事情。因為大圖的原因,我們會在Crash報告中看到OOM(記憶體不足).Android的記憶體有限,這一點我們應該心裡有數。
stackoverflow上有許多相關問題的回答,當你碰到oom時,可以直接跳過本文,貼上複製答案即可。但是對於其他人來說,我想告訴你們一些載入大圖的知識和原理。
載入Bitmap到記憶體
so easy.你所需要做的就是使用BitmapFactory解碼你的圖片。
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage); imageView.setImageBitmap(bitmap);
看起來好像完事啦麼?我要告訴你這裡有個問題。讓我們來看看解碼後的圖片佔用的記憶體大小。
bitmap.getByteCount()可以返回圖片的大小。其記憶體大小為12262248 位元組,等於12.3MB .沒錯,你可能會有些疑惑。圖片在磁碟上的大小為3.5MB ,getByteCount()得出的結果卻大於3.5MB。原因如下:
圖片以JPG,PNG或者其他格式壓縮後儲存在計算機上。一旦你將圖片載入到記憶體後,圖片將不會唄壓縮,而是佔用所有畫素所需的記憶體。
步驟
- 不用載入圖片到記憶體獲取其尺寸
- 通過圖片尺寸計算縮放係數
- 通過計算後值載入圖片到記憶體
BitmapFactory.Options
這個類是個元資料提供器,我們可以通過這個類來完成第一步。
BitmapFactory.Options options = new BitmapFactory.Options(); options.inJustDecodeBounds = true; BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage, options);
我們傳BitmapFactory.Options例項到BitmapFactory.decodeSource() 方法。我們將inJustDecodeBounds 設定為true .inJustDecodeBounds是什麼意思呢?就是我們不用講圖片載入到記憶體中。我們只想得到關於圖片的資訊。我們可以通過這個資訊計算縮放係數。
執行此段程式碼得到如下資訊:
options.outHeight : 1126 options.outWidth : 2000 options.bitmap : null
減少圖片大小(記憶體中)
現在該計算inSampleSize 啦,等等,何為inSampleSize?inSampleSize是屬於 BitmapFactory.Option這個類的一個縮放係數。
如果我們有一張10001000的圖片,inSampleSize為2,解碼後大小為500 500.
BitmapFactory.Options options = new BitmapFactory.Options(); options.inJustDecodeBounds = true; options.inSampleSize = 3; BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage, options);
我們可以這樣玩麼?不行。因為我們不知道圖片大小,如果是小圖,圖片將會變得更小,我們的使用者只能看到一些畫素而不是圖片。有些圖片將會縮放5倍,有些則為2倍。我們將縮放係數定義為一個常數。所以我們需要根據圖片尺寸來計算。
計算inSampleSize的方法取決於你,我的意思就是根據你的需要來設計自己的演算法。Android官方文件中,是基於2的冪次方來計算inSampleSize的。
options.inSampleSize = calculateInSampleSize(options, 500,500); options.inJustDecodeBounds = false; Bitmap smallBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.hqimage, options);
我們將inJustDecodeBounds變為false,以便獲取bitmap.
現在,bitmap.getByteCount() 返回的大小為3.1MB。這個是記憶體大小。
圖片的記憶體大小由12.3MB變為3.1MB,記憶體大小減少啦75%。
減少圖片尺寸(磁碟上)
我們也可以減少圖片在磁碟上的大小。我們可以通過Bitmap的compress方法來壓縮圖片。
讓我們來看下不改邊圖片質量的尺寸大小,100表示質量不變
ByteArrayOutputStream bos = new ByteArrayOutputStream(); bitmap.compress(Bitmap.CompressFormat.JPEG, 100, bos); byte[] bitmapdata = bos.toByteArray();
原圖尺寸為1.6MB,改變圖片質量後文件大小將發生何種變化?
bitmap.compress(Bitmap.CompressFormat.JPEG, 50, bos);
圖片質量改為50,大小變成24.4KB
如果你想改變bitmap的質量,壓縮格式必須為.JPEG ,PNG格式的圖片質量是不會發生變化的。
檔案大小由1.6MB變成啦24.4KB。