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