在android開發(fā)中,圖片的處理是一個android程序員的必備技能,不少初學(xué)者對圖片的處理還是有些迷惑,為什么加載一個3MB的圖片會出現(xiàn)OOM,到底如何處理才能避免這種情況。下面我就針對這個問題做一個簡單的講解。首先我們應(yīng)該知道的是Android系統(tǒng)為一個應(yīng)用程序分配的內(nèi)存空間是有限的,可以通過以下的代碼得到當(dāng)前系統(tǒng)為我們的應(yīng)用程序分配的空間(一些游戲需要的內(nèi)存比較大,它可能會通過代碼改動這個值)。
int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024/1024);
好了現(xiàn)在知道你應(yīng)用程序所能占用的最大內(nèi)存了,再來看一下一個3MB的圖片加載進內(nèi)存所占用的空間是多少,首先我找了一個3MB左右的圖片,他的像素是3240*2160,差不多是800萬像素鏡頭拍出的,把他加載到內(nèi)存中時,我們可以指定圖片的壓縮質(zhì)量參數(shù),Config.ARGB_4444,Config.ARGB_8888,Config.RGB_565,他們表示色彩的存儲方式,在繪制一個像素點時,所占用的字節(jié)數(shù)也會有所不同,分別為16字節(jié),32字節(jié),16字節(jié),假如我們圖片壓縮質(zhì)量參數(shù)為Config.RGB_565,那么該圖片所占用的空間為3240*2160*16/1024/1024(MB),大約是106兆,跟上面咱們得出的一個應(yīng)用程序所分配的最大空間對比,看是不是圖片所占空間導(dǎo)致了OOM。
大家都知道,目前我們主流屏幕的像素是1080*720,我們不可能將一個3240*2160的圖片中所有的像素點在當(dāng)前屏幕上顯示出來,這時就需要我們改變圖片的大小,與我們屏幕的像素密度相當(dāng),這時既減少了圖片所占用的內(nèi)存,也不會造成圖片的失真。
首先我們需要創(chuàng)建一個Bitmap對象,BitmapFactory這個類提供了多個方法來創(chuàng)建Bitmap對象,根據(jù)圖片的來源選擇合適的方法來創(chuàng)建對象。創(chuàng)建對象時,會為Bitmap對象分配內(nèi)存,這時很容易出現(xiàn)OOM,我們就需要一個BitmapFactory.Options參數(shù)(Options一般為參數(shù)配置類),通過這個參數(shù),我們指定將這個參數(shù)的inJustDecodeBounds屬性設(shè)置為true,就可以讓解析方法禁止為bitmap分配內(nèi)存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null,但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值,通過這種方式我們就得到了圖片的長寬,這是第一次取樣;然后根據(jù)我們需要的圖片寬高得到一個壓縮比例,設(shè)置到options.inSampleSize屬性上,同時options.inJustDecodeBound設(shè)置為false,在options上還可以設(shè)置解析圖片的壓縮質(zhì)量參數(shù),之后繼續(xù)調(diào)用解析方式進行解析,這一次得到的bitmap對象就是一個按比例壓縮后的圖片了。
BitmapFactory.Options opts = new Options();
// 設(shè)置不去真正的解析Bitmap,只解析到其邊界信息
opts.inJustDecodeBounds = true;
Bitmap bm = BitmapFactory.decodeFile(Environment
.getExternalStorageDirectory().toString() + "/2.jpg", opts);
int imageWidth = opts.outWidth;
int imageHeight = opts.outHeight;
// 計算縮放的比例
int scale = 0;
int scaleX = imageWidth / width;
int scaleY = imageHeight / height;
if (scaleX > scaleY && scaleX > 1) {
scale = scaleX;
}
if (scaleY > scaleX && scaleY > 1) {
scale = scaleY;
}
//設(shè)置真正的開始解析Bitmap
opts.inJustDecodeBounds = false;
// 設(shè)置縮放比例
opts.inSampleSize = scale;
//設(shè)置圖片的壓縮質(zhì)量參數(shù)
opts.inPreferredConfig=Config.RGB_565;
//開始解析
bm = BitmapFactory.decodeFile(Environment.getExternalStorageDirectory()
.toString() + "/2.jpg", opts);
//將圖片設(shè)置到ImageView上
iv.setImageBitmap(bm);