Android加載大分辨率圖片到手機(jī)內(nèi)存中的實(shí)例方法
還原堆內(nèi)存溢出的錯(cuò)誤
首先來還原一下堆內(nèi)存溢出的錯(cuò)誤。首先在SD卡上放一張照片,分辨率為(3776 X 2520),大小為3.88MB,是我自己用相機(jī)拍的一張照片。應(yīng)用的布局很簡單,一個(gè)Button一個(gè)ImageView,然后按照常規(guī)的方式,使用BitmapFactory加載一張照片并使用一個(gè)ImageView展示。
代碼如下:
btn_loadimage.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Bitmap bitmap=BitmapFactory.decodeFile("/sdcard/a.jpg");
iv_bigimage.setImageBitmap(bitmap);
}
}
當(dāng)點(diǎn)擊按鈕后,程序會報(bào)錯(cuò),查看日志為:
先來分析一下這個(gè)錯(cuò)誤,首先dalvikvm(Android虛擬機(jī))發(fā)現(xiàn)需要的內(nèi)存38MB大于應(yīng)用的堆內(nèi)存24MB,這個(gè)時(shí)候嘗試使用軟加載的方式加載數(shù)據(jù),我們知道當(dāng)內(nèi)存不足的時(shí)候dalvikvm會自動(dòng)進(jìn)行GC(Garbage Collection),大概清理了55k的空間出來,耗時(shí)203毫秒,但是內(nèi)存還是不夠,所以最后發(fā)生堆內(nèi)存溢出的錯(cuò)誤。
分析堆內(nèi)存溢出
Android系統(tǒng)主要用于低能耗的移動(dòng)設(shè)備,所以對內(nèi)存的管理有很多限制,一個(gè)應(yīng)用程序,Android系統(tǒng)缺省會為其分配最大16MB(某些機(jī)型是24MB)的空間作為堆內(nèi)存空間,我這里使用的模擬器調(diào)試的,這個(gè)模擬器被設(shè)定為24MB,可以在Android Virtual Device Manager中查看到。
而這里的圖片明明只有3.88MB,遠(yuǎn)遠(yuǎn)小于Android為應(yīng)用分配的堆內(nèi)存,而加載到內(nèi)存中,為什么需要消耗大約38MB的內(nèi)存呢?
我們都知道,圖片是由一個(gè)一個(gè)點(diǎn)分布組成的(分辨率),通常加載這類數(shù)據(jù)都會在內(nèi)存中創(chuàng)建一個(gè)二維數(shù)組,數(shù)組中的每一項(xiàng)代表一個(gè)點(diǎn),而這個(gè)圖片的分辨率是3776 * 2520,每一點(diǎn)又是由ARGB色組成,每個(gè)色素占4個(gè)Byte,所以這張圖片加載到內(nèi)存中需要消耗的內(nèi)存為:
3776 * 2520 * 4byte = 38062080byte
大約需要38MB的內(nèi)存才能正確加載這張圖片,這就是上面錯(cuò)誤描述需要38MB的內(nèi)存空間,大小略有出入,因?yàn)閳D片還有一些Exif信息需要存儲,會比僅靠分辨率計(jì)算要大一些。
如何加載大分辨率圖片
有時(shí)候我們確實(shí)會需要加載一些大分辨率的圖片,但是對于移動(dòng)設(shè)備而言,哪怕加載能成功那么大的內(nèi)存也是一種浪費(fèi)(屏幕分辨率限制),所以就需要想辦法把圖片按照一定比率壓縮,使分辨率降低,以至于又不需要耗費(fèi)很大的堆內(nèi)存空間,又可以最大的利用設(shè)備屏幕的分辨率來顯示圖片。這里就用到一個(gè)BitmapFactory.Options對象,下面來介紹它。
BitmapFactory.Options為BitmapFactory的一個(gè)內(nèi)部類,它主要用于設(shè)定與存儲BitmapFactory加載圖片的一些信息。下面是Options中需要用到的屬性:
inJustDecodeBounds:如果設(shè)置為true,將不把圖片的像素?cái)?shù)組加載到內(nèi)存中,僅加載一些額外的數(shù)據(jù)到Options中。
outHeight:圖片的高度。
outWidth:圖片的寬度。
inSampleSize:如果設(shè)置,圖片將依據(jù)此采樣率進(jìn)行加載,不能設(shè)置為小于1的數(shù)。例如設(shè)置為4,分辨率寬和高將為原來的1/4,這個(gè)時(shí)候整體所占內(nèi)存將是原來的1/16。
示例Demo
下面通過一個(gè)簡單的Demo來演示上面提到的內(nèi)容,代碼中注釋比較清晰,這里就不再累述了。
package cn.bgxt.loadbigimg;
import android.os.Bundle;
import android.os.Environment;
import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.BitmapFactory.Options;
import android.view.Menu;
import android.view.View;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.ImageView;
public class MainActivity extends Activity {
private Button btn_loadimage;
private ImageView iv_bigimage;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btn_loadimage = (Button) findViewById(R.id.btn_loadimage);
iv_bigimage = (ImageView) findViewById(R.id.iv_bigimage);
btn_loadimage.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// Bitmap bitmap=BitmapFactory.decodeFile("/sdcard/a.jpg");
// iv_bigimage.setImageBitmap(bitmap);
BitmapFactory.Options opts = new Options();
// 不讀取像素?cái)?shù)組到內(nèi)存中,僅讀取圖片的信息
opts.inJustDecodeBounds = true;
BitmapFactory.decodeFile("/sdcard/a.jpg", opts);
// 從Options中獲取圖片的分辨率
int imageHeight = opts.outHeight;
int imageWidth = opts.outWidth;
// 獲取Android屏幕的服務(wù)
WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
// 獲取屏幕的分辨率,getHeight()、getWidth已經(jīng)被廢棄掉了
// 應(yīng)該使用getSize(),但是這里為了向下兼容所以依然使用它們
int windowHeight = wm.getDefaultDisplay().getHeight();
int windowWidth = wm.getDefaultDisplay().getWidth();
// 計(jì)算采樣率
int scaleX = imageWidth / windowWidth;
int scaleY = imageHeight / windowHeight;
int scale = 1;
// 采樣率依照最大的方向?yàn)闇?zhǔn)
if (scaleX > scaleY && scaleY >= 1) {
scale = scaleX;
}
if (scaleX < scaleY && scaleX >= 1) {
scale = scaleY;
}
// false表示讀取圖片像素?cái)?shù)組到內(nèi)存中,依照設(shè)定的采樣率
opts.inJustDecodeBounds = false;
// 采樣率
opts.inSampleSize = scale;
Bitmap bitmap = BitmapFactory.decodeFile("/sdcard/a.jpg", opts);
iv_bigimage.setImageBitmap(bitmap);
}
});
}
}
效果展示:
總結(jié)
這里講解了如何加載一個(gè)大分辨率的圖片到內(nèi)存中并使用它。不過一般好一點(diǎn)的圖片處理軟件,都會有圖片放大功能,如果僅做此處理,單純的把處理后的圖片放大,會影響顯示效果,圖片還原度不高。一般會重新獲取放大區(qū)域的圖片的分辨率像素?cái)?shù)組,然后重新處理加載到內(nèi)存中進(jìn)行顯示。
- Android中Glide加載庫的圖片緩存配置究極指南
- android異步加載圖片并緩存到本地實(shí)現(xiàn)方法
- Android中ListView異步加載圖片錯(cuò)位、重復(fù)、閃爍問題分析及解決方案
- Android關(guān)于Glide的使用(高斯模糊、加載監(jiān)聽、圓角圖片)
- Android使用控件ImageView加載圖片的方法
- Android圖片加載利器之Picasso基本用法
- Android實(shí)現(xiàn)加載廣告圖片和倒計(jì)時(shí)的開屏布局
- Android 加載GIF圖最佳實(shí)踐方案
- android中Glide實(shí)現(xiàn)加載圖片保存至本地并加載回調(diào)監(jiān)聽
- 使用Thumbnails實(shí)現(xiàn)圖片指定大小壓縮
相關(guān)文章
Android view更改背景資源與padding消失的問題解決辦法
這篇文章主要介紹了Android view更改背景資源與padding消失的問題解決辦法的相關(guān)資料,需要的朋友可以參考下2017-04-04
AndroidStudio3.6.1打包jar及AndroidStudio4.0打包jar的一系列問題及用法
這篇文章主要介紹了AndroidStudio3.6.1打包jar,AndroidStudio4.0打包jar的問題及用法,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
AndroidStudio 3.6 中 R.layout 找不到對應(yīng)的xml文件問題及解決方法
這篇文章主要介紹了AndroidStudio 3.6 中 R.layout 找不到對應(yīng)的xml文件問題,本文給出了解決方法對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
Android?使用flow實(shí)現(xiàn)倒計(jì)時(shí)的方式
這篇文章主要介紹了Android?使用flow實(shí)現(xiàn)倒計(jì)時(shí)的方式,借助Flow這個(gè)工具,更加優(yōu)雅地實(shí)現(xiàn)這個(gè)需求功能,文末給大家整理了Android?實(shí)現(xiàn)倒計(jì)時(shí)的幾種方式,需要的朋友可以參考下2022-04-04
Android TabLayout(選項(xiàng)卡布局)簡單用法實(shí)例分析
這篇文章主要介紹了Android TabLayout(選項(xiàng)卡布局)簡單用法,結(jié)合實(shí)例形式簡單分析了Android選項(xiàng)卡布局的界面布局與功能實(shí)現(xiàn)具體相關(guān)技巧,需要的朋友可以參考下2016-01-01
android實(shí)現(xiàn)動(dòng)態(tài)顯隱進(jìn)度條
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)動(dòng)態(tài)顯隱進(jìn)度條,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-07-07
Android實(shí)現(xiàn)在map上畫出路線的方法
這篇文章主要介紹了Android實(shí)現(xiàn)在map上畫出路線的方法,較為詳細(xì)的分析了Android在map上繪制路線所涉及的map圖調(diào)用、畫筆的使用、頁面布局及權(quán)限控制的相關(guān)技巧,需要的朋友可以參考下2015-07-07
談?wù)凙ndroid里的Context的使用實(shí)例
這篇文章主要介紹了談?wù)凙ndroid里的Context的使用實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。2016-11-11
Android編程實(shí)現(xiàn)canvas繪制柱狀統(tǒng)計(jì)圖功能【自動(dòng)計(jì)算寬高及分度值、可左右滑動(dòng)】
這篇文章主要介紹了Android編程實(shí)現(xiàn)canvas繪制柱狀統(tǒng)計(jì)圖功能,具備自動(dòng)計(jì)算寬高及分度值及左右滑動(dòng)的功能,涉及Android canvas繪圖操作相關(guān)技巧,需要的朋友可以參考下2017-01-01

