這篇文章主要介紹“android大圖片加載OOM處理問題怎么解決”的相關(guān)知識,小編通過實際案例向大家展示操作過程,操作方法簡單快捷,實用性強,希望這篇“android大圖片加載OOM處理問題怎么解決”文章能幫助大家解決問題。
目前創(chuàng)新互聯(lián)已為上千家的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)頁空間、網(wǎng)站托管維護、企業(yè)網(wǎng)站設(shè)計、章貢網(wǎng)站維護等服務(wù),公司將堅持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長,共同發(fā)展。
我們在編寫Android程序的時候經(jīng)常要用到許多圖片,不同圖片總是會有不同的形狀、不同的大小,但在大多數(shù)情況下,這些圖片都會大于我們程序所需要的大小。比如說系統(tǒng)圖片庫里展示的圖片大都是用手機攝像頭拍出來的,這些圖片的分辨率會比我們手機屏幕的分辨率高得多。大家應(yīng)該知道,我們編寫的應(yīng)用程序都是有一定內(nèi)存限制的,程序占用了過高的內(nèi)存就容易出現(xiàn)OOM(OutOfMemory)異常。
先看一個栗子:主要功能為:ListView 的滑動列表展示,每個條目是一張圖片
xml主頁面布局
ListView 的item 布局
布局界面非常簡單,不用多說什么,java代碼
public class MainActivity extends AppCompatActivity @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); ListView listView = (ListView) findViewById(R.id.listView); listView.setAdapter(new MyAdapter()); } class MyAdapter extends BaseAdapter { @Override public int getCount() { return 50; } @Override public Object getItem(int position) { return null; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { View view = View.inflate(MainActivity.this, R.layout.list_item, null); ImageView imageView = (ImageView) view.findViewById(R.id.imageView); imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異常卡頓 return
由于我們測試的圖片非常大,屬于超清的那種,我們用默認的imageView.setImageResource(R.drawable.girl)方法,是完全顯示此高清圖片,此時
在模擬器上運行之后,雖然能夠顯示,但是界面灰??D有木有,可以說根本劃不動的,口說無憑,上圖
唉!不好意思沒圖了,本來已經(jīng)錄好圖了,奈何超過2M沒法上傳,總之就是很卡很卡啦
看看,不騙人,真的很卡呀!!!
然后,接下來我們就要解決這個問題了,怎么解決呢,我們通過思考就會發(fā)現(xiàn),其實造成這種結(jié)果單位主要原因就是要顯示的圖片太大了,我們完全不需要展示這麼大的圖片,沒錯我們可以把圖片壓縮一下,比如壓縮成 100*100 的啦.
我們主要使用的就是BitmapFactory.decodeResource()方法,該方法有多個重載,我們使用
decodeResource (Resources res,
int id,
BitmapFactory.Options opts)(該方法返回一個Bitmap對象.)
和 BitmapFactory.Options() 這個類該類可以對要加載的圖片進行配置
下面我給出代碼詳細注釋:
首先創(chuàng)建一個方法 decodeSampleBitmapFromResource() 主要來把大圖片解析為適當?shù)膱D片
public static Bitmap decodeSampleBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) { // 第一次解析將inJustDecodeBounds設(shè)置為true,來獲取圖片大小 BitmapFactory.Options options = new BitmapFactory.Options(); //將這個參數(shù)的inJustDecodeBounds屬性設(shè)置為true就可以讓解析方法禁止為bitmap分配內(nèi)存 options.inJustDecodeBounds = true; /*解析源圖片,返回一個 bitmap 對象,當 options.inJustDecodeBounds = true; 禁止為bitmap分配內(nèi)存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null了, 但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值。 這個技巧讓我們可以在加載圖片之前就獲取到圖片的長寬值和MIME類型,從而根據(jù)情況對圖片進行壓縮*/ BitmapFactory.decodeResource(res, resId, options); // 這個方法用來計算inSampleSize值 options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight); // 使用獲取到的inSampleSize值再次解析圖片 options.inJustDecodeBounds = false; /*計算完inSampleSize 的合適大小后,需要把 options.inJustDecodeBounds = false; 然后把再 BitmapFactory.decodeResource(res,resId,options) 此時 options.inJustDecodeBounds = false; ,BitmapFactory.decodeResource() 方法返回一個bitmap對象給 imageView.setImageBitmap()方法 從而顯示一個合適大小的圖片 */ return
calculateInSampleSize(options, reqWidth, reqHeight); 的實現(xiàn)
/*計算出合適的inSampleSize值:*/ public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) { //源圖片的高度和寬度 final int height = options.outHeight;//得到要加載的圖片高度 final int width = options.outWidth; int inSampleSize = 1; if (height > reqHeight || width > reqWidth) { //計算出實際寬高和目標寬高的比例 final int heightRatio = Math.round((float) height / (float) reqHeight); final int widthRatio = Math.round((float) width / (float) reqWidth); // 選擇寬和高中最小的比率作為inSampleSize的值,這樣可以保證最終圖片的寬和高 // 一定都會大于等于目標的寬和高。 inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio; } return
上面是對用到的兩個自定義的方法的解釋,把他倆用到 主Java 代碼中如下:
最后給出java完整程序
public class MainActivity extends AppCompatActivity @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); ListView listView = (ListView) findViewById(R.id.listView); listView.setAdapter(new MyAdapter()); } class MyAdapter extends BaseAdapter { @Override public int getCount() { return 50; } @Override public Object getItem(int position) { return null; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { View view = View.inflate(MainActivity.this, R.layout.list_item, null); ImageView imageView = (ImageView) view.findViewById(R.id.imageView); // imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異常卡頓 //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小 imageView.setImageBitmap(decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100)); return view; } } /*計算出合適的inSampleSize值:*/ public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) { //源圖片的高度和寬度 final int height = options.outHeight;//得到要加載的圖片高度 final int width = options.outWidth; int inSampleSize = 1; if (height > reqHeight || width > reqWidth) { //計算出實際寬高和目標寬高的比例 final int heightRatio = Math.round((float) height / (float) reqHeight); final int widthRatio = Math.round((float) width / (float) reqWidth); // 選擇寬和高中最小的比率作為inSampleSize的值,這樣可以保證最終圖片的寬和高 // 一定都會大于等于目標的寬和高。 inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio; } return inSampleSize; } public static Bitmap decodeSampleBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) { // 第一次解析將inJustDecodeBounds設(shè)置為true,來獲取圖片大小 BitmapFactory.Options options = new BitmapFactory.Options(); //將這個參數(shù)的inJustDecodeBounds屬性設(shè)置為true就可以讓解析方法禁止為bitmap分配內(nèi)存 options.inJustDecodeBounds = true; //解析源圖片,返回一個 bitmap 對象,當 options.inJustDecodeBounds = true; /*禁止為bitmap分配內(nèi)存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null了, 但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值。 這個技巧讓我們可以在加載圖片之前就獲取到圖片的長寬值和MIME類型,從而根據(jù)情況對圖片進行壓縮*/ BitmapFactory.decodeResource(res, resId, options); // 調(diào)用上面定義的方法計算inSampleSize值 options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight); // 使用獲取到的inSampleSize值再次解析圖片 options.inJustDecodeBounds = false; /*計算完inSampleSize 的合適大小后,需要把 options.inJustDecodeBounds = false; 然后把再 BitmapFactory.decodeResource(res,resId,options) 此時 options.inJustDecodeBounds = false; ,BitmapFactory.decodeResource() 方法返回一個bitmap對象給 imageView.setImageBitmap()方法 從而顯示一個合適大小的圖片 */ return
我們把 imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異??D
這行代碼,改為這樣寫
//對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小 imageView.setImageBitmap(decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100));
在我快速的來回滑動當中,然后就
并且報了下面的異常信息
很明顯的OOM了有木有,那我們又該用啥辦法解決呢,仔細思考我們有發(fā)現(xiàn)了問題的關(guān)鍵所在,就是ListView 條目的圖片快速加載,快速釋放,在某一時刻,圖片加載過多,導(dǎo)致了 OOM 的問題.我們自然就可以想到,能不能別讓滾出屏幕的圖片離開呢,是不是可以把他們緩存起來呢,
所以,如果朋友們熟悉LruCache 這個類的話,又會想到解決方案了,不熟悉的也沒關(guān)系,我還是會給出詳細注釋的:
首先看 我們自定義的 initLruCache(); 方法,在 onCreate() 方法中調(diào)用
/** * 初始化緩存 */ private void initLruCache() { // 獲取到可用內(nèi)存的最大值,使用內(nèi)存超出這個值會引起OutOfMemory異常。 // LruCache通過構(gòu)造函數(shù)傳入緩存值,以KB為單位。 int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024); // 使用最大可用內(nèi)存值的1/8作為緩存的大小。 int cacheSize = maxMemory / 8; mMemoryCache = new LruCache(cacheSize) { @Override protected int sizeOf(String key, Bitmap bitmap) { // 重寫此方法來衡量每張圖片的大小,默認返回圖片數(shù)量。 return bitmap.getByteCount() / 1024; } }; }
然后再ListView 的適配器中的 getView() 方法中給ImageView 設(shè)置圖片時,吧圖片從緩存中取出,放入就行了
//得到資源id的唯一標示 Bitmap bitmap = mMemoryCache.get(imageKey);//從lruCache中取出該緩存 //如果緩存中有該資源 if (bitmap != null) { imageView.setImageBitmap(bitmap); } else { //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小 Bitmap bitmap2 = decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100); imageView.setImageBitmap(bitmap2); mMemoryCache.put(imageKey, bitmap2);//添加到緩存中
完整java 代碼如下
public class MainActivity extends AppCompatActivity private LruCachemMemoryCache; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initLruCache(); ListView listView = (ListView) findViewById(R.id.listView); listView.setAdapter(new MyAdapter()); } /** * 初始化緩存 */ private void initLruCache() { // 獲取到可用內(nèi)存的最大值,使用內(nèi)存超出這個值會引起OutOfMemory異常。 // LruCache通過構(gòu)造函數(shù)傳入緩存值,以KB為單位。 int maxMemory = (int) (Runtime.getRuntime().maxMemory() / 1024); // 使用最大可用內(nèi)存值的1/8作為緩存的大小。 int cacheSize = maxMemory / 8; mMemoryCache = new LruCache (cacheSize) { @Override protected int sizeOf(String key, Bitmap bitmap) { // 重寫此方法來衡量每張圖片的大小,默認返回圖片數(shù)量。 return bitmap.getByteCount() / 1024; } }; } class MyAdapter extends BaseAdapter { @Override public int getCount() { return 50; } @Override public Object getItem(int position) { return null; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View convertView, ViewGroup parent) { View view = View.inflate(MainActivity.this, R.layout.list_item, null); ImageView imageView = (ImageView) view.findViewById(R.id.imageView); // imageView.setImageResource(R.drawable.girl);//完全顯示圖片大小,ListView 滑動界面異??D String imageKey = String.valueOf(R.drawable.girl);//得到資源id的唯一標示 Bitmap bitmap = mMemoryCache.get(imageKey);//從lruCache中取出該緩存 //如果緩存中有該資源 if (bitmap != null) { imageView.setImageBitmap(bitmap); } else { //對非常大的圖片進行壓縮顯示(壓縮成100*100顯示),以適合的寬高顯示大小 Bitmap bitmap2 = decodeSampleBitmapFromResource(getResources(), R.drawable.girl, 100, 100); imageView.setImageBitmap(bitmap2); mMemoryCache.put(imageKey, bitmap2);//添加到緩存中 } return view; } } /*計算出合適的inSampleSize值:*/ public static int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) { //源圖片的高度和寬度 final int height = options.outHeight;//得到要加載的圖片高度 final int width = options.outWidth; int inSampleSize = 1; if (height > reqHeight || width > reqWidth) { //計算出實際寬高和目標寬高的比例 final int heightRatio = Math.round((float) height / (float) reqHeight); final int widthRatio = Math.round((float) width / (float) reqWidth); // 選擇寬和高中最小的比率作為inSampleSize的值,這樣可以保證最終圖片的寬和高 // 一定都會大于等于目標的寬和高。 inSampleSize = heightRatio < widthRatio ? heightRatio : widthRatio; } return inSampleSize; } public static Bitmap decodeSampleBitmapFromResource(Resources res, int resId, int reqWidth, int reqHeight) { // 第一次解析將inJustDecodeBounds設(shè)置為true,來獲取圖片大小 BitmapFactory.Options options = new BitmapFactory.Options(); //將這個參數(shù)的inJustDecodeBounds屬性設(shè)置為true就可以讓解析方法禁止為bitmap分配內(nèi)存 options.inJustDecodeBounds = true; //解析源圖片,返回一個 bitmap 對象,當 options.inJustDecodeBounds = true; /*禁止為bitmap分配內(nèi)存,返回值也不再是一個Bitmap對象,而是null。雖然Bitmap是null了, 但是BitmapFactory.Options的outWidth、outHeight和outMimeType屬性都會被賦值。 這個技巧讓我們可以在加載圖片之前就獲取到圖片的長寬值和MIME類型,從而根據(jù)情況對圖片進行壓縮*/ BitmapFactory.decodeResource(res, resId, options); // 調(diào)用上面定義的方法計算inSampleSize值 options.inSampleSize = calculateInSampleSize(options, reqWidth, reqHeight); // 使用獲取到的inSampleSize值再次解析圖片 options.inJustDecodeBounds = false; /*計算完inSampleSize 的合適大小后,需要把 options.inJustDecodeBounds = false; 然后把再 BitmapFactory.decodeResource(res,resId,options) 此時 options.inJustDecodeBounds = false; ,BitmapFactory.decodeResource() 方法返回一個bitmap對象給 imageView.setImageBitmap()方法 從而顯示一個合適大小的圖片 */ return
關(guān)于“android大圖片加載OOM處理問題怎么解決”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識,可以關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,小編每天都會為大家更新不同的知識點。