`

避免出现bitmap内存限制OUT OF MEMORY的一种方法

阅读更多
在编写Android程序的时候,我们总是难免会碰到OOM(OUT OF MEMORY)的错误,那么这个错误究竟是怎么来的呢,可以先看一下这篇文章ANDROID BITMAP内存限制OOM,OUT OF MEMORY。



这里,我使用Gallery来举例,在模拟器中,不会出现OOM错误,但是,一旦把程序运行到真机里,图片文件一多,必然会出现OOM,我们通过做一些额外的处理来避免。

1.创建一个图片缓存对象HashMap<Integer,Bitmap> dataCache,integer对应Adapter中的位置position,我们只用缓存处在显示中的图片,对于之外的位置,如果dataCache中有对应的图片,我们需要进行回收内存。在这个例子中,Adapter对象的getView方法首先判断该位置是否有缓存的bitmap,如果没有,则解码图片(bitmapDecoder.getPhotoItem,BitmapDecoder类见后面)并返回bitmap对象,设置dataCache在该位置上的bitmap缓存以便之后使用;若是该位置存在缓存,则直接取出来使用,避免了再一次调用底层的解码图像需要的内存开销。有时为了提高Gallery的更新速度,我们还可以预存储一些位置上的bitmap,比如存储显示区域位置外向上3个向下3个位置的bitmap,这样上或下滚动Gallery时可以加快getView的获取。

Java代码 
public View getView(int position, View convertView, ViewGroup parent) {   
       
    if(convertView==null){   
        LayoutInflater inflater  = LayoutInflater.from(context);   
        convertView = inflater.inflate(R.layout.photo_item, null);   
  
           holder = new ViewHolder();   
           holder.photo = (ImageView) convertView.findViewById(R.id.photo_item_image);   
           holder.photoTitle = (TextView) convertView.findViewById(R.id.photo_item_title);   
           holder.photoDate = (TextView) convertView.findViewById(R.id.photo_item_date);   
           convertView.setTag(holder);   
    }else {   
       holder = (ViewHolder) convertView.getTag();   
    }   
    cursor.moveToPosition(position);   
       
    Bitmap current = dateCache.get(position);   
    if(current != null){//如果缓存中已解码该图片,则直接返回缓存中的图片   
        holder.photo.setImageBitmap(current);   
    }else {   
        current = bitmapDecoder.getPhotoItem(cursor.getString(1), 2) ;   
        holder.photo.setImageBitmap(current);   
        dateCache.put(position, current);   
    }   
    holder.photoTitle.setText(cursor.getString(2));   
    holder.photoDate.setText(cursor.getString(4));   
    return convertView;   
}   
  

		public View getView(int position, View convertView, ViewGroup parent) {
			
			if(convertView==null){
				LayoutInflater inflater  = LayoutInflater.from(context);
				convertView = inflater.inflate(R.layout.photo_item, null);

	            holder = new ViewHolder();
	            holder.photo = (ImageView) convertView.findViewById(R.id.photo_item_image);
	            holder.photoTitle = (TextView) convertView.findViewById(R.id.photo_item_title);
	            holder.photoDate = (TextView) convertView.findViewById(R.id.photo_item_date);
	            convertView.setTag(holder);
			}else {
		       holder = (ViewHolder) convertView.getTag();
		    }
			cursor.moveToPosition(position);
			
			Bitmap current = dateCache.get(position);
			if(current != null){//如果缓存中已解码该图片,则直接返回缓存中的图片
				holder.photo.setImageBitmap(current);
			}else {
				current = bitmapDecoder.getPhotoItem(cursor.getString(1), 2) ;
				holder.photo.setImageBitmap(current);
				dateCache.put(position, current);
			}
			holder.photoTitle.setText(cursor.getString(2));
			holder.photoDate.setText(cursor.getString(4));
			return convertView;
		}
		
	}
BitmapDecoder.class


Java代码 
package com.wuyi.bestjoy;   
  
import java.io.FileNotFoundException;   
import java.io.FileOutputStream;   
  
import android.content.Context;   
import android.graphics.Bitmap;   
import android.graphics.BitmapFactory;   
import android.graphics.Matrix;   
  
public class BitmapDecoder {   
    private static final String TAG = "BitmapDecoder";   
    private Context context;   
    public BitmapDecoder(Context context) {   
        this.context = context;   
    }   
       
    public Bitmap getPhotoItem(String filepath,int size) {   
          BitmapFactory.Options options = new BitmapFactory.Options();   
          options.inSampleSize=size;   
          Bitmap bitmap = BitmapFactory.decodeFile(filepath,options);   
          bitmap=Bitmap.createScaledBitmap(bitmap, 180, 251, true);//预先缩放,避免实时缩放,可以提高更新率   
          return bitmap;   
           
    }   
}  

package com.wuyi.bestjoy;

import java.io.FileNotFoundException;
import java.io.FileOutputStream;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Matrix;

public class BitmapDecoder {
	private static final String TAG = "BitmapDecoder";
	private Context context;
	public BitmapDecoder(Context context) {
		this.context = context;
	}
	
	public Bitmap getPhotoItem(String filepath,int size) {
	      BitmapFactory.Options options = new BitmapFactory.Options();
  	      options.inSampleSize=size;
  	      Bitmap bitmap = BitmapFactory.decodeFile(filepath,options);
  	      bitmap=Bitmap.createScaledBitmap(bitmap, 180, 251, true);//预先缩放,避免实时缩放,可以提高更新率
	      return bitmap;
  	    
	}
}


2.由于Gallery控件的特点,总有一个item处于当前选择状态,我们利用此时进行dataCache中额外不用的bitmap的清理,来释放内存。


Java代码 
@Override  
    public void onItemSelected(AdapterView<?> parent, View view, int position,long id) {   
           
        releaseBitmap();   
        Log.v(TAG, "select id:"+ id);   
    }   
  
private void releaseBitmap(){   
    //在这,我们分别预存储了第一个和最后一个可见位置之外的3个位置的bitmap   
    //即dataCache中始终只缓存了(M=6+Gallery当前可见view的个数)M个bitmap   
        int start = mGallery.getFirstVisiblePosition()-3;   
        int end = mGallery.getLastVisiblePosition()+3;   
        Log.v(TAG, "start:"+ start);   
        Log.v(TAG, "end:"+ end);   
        //释放position<start之外的bitmap资源   
        Bitmap delBitmap;   
        for(int del=0;del<start;del++){   
            delBitmap = dateCache.get(del);   
            if(delBitmap != null){   
                                //如果非空则表示有缓存的bitmap,需要清理   
                Log.v(TAG, "release position:"+ del);   
                //从缓存中移除该del->bitmap的映射   
                                dateCache.remove(del);   
                delBitmap.recycle();   
            }   
        }   
  
        freeBitmapFromIndex(end);   
           
    }   
       
    /**  
     * 从某一位置开始释放bitmap资源  
     * @param index  
     */  
    private void freeBitmapFromIndex(int end) {   
        //释放之外的bitmap资源   
        Bitmap delBitmap;   
        for(int del =end+1;del<dateCache.size();del++){   
            delBitmap = dateCache.get(del);   
            if(delBitmap != null){   
                dateCache.remove(del);   
                delBitmap.recycle();   
                Log.v(TAG, "release position:"+ del);   
            }   
               
        }   
    }  

@Override
	public void onItemSelected(AdapterView<?> parent, View view, int position,long id) {
		
		releaseBitmap();
		Log.v(TAG, "select id:"+ id);
	}

private void releaseBitmap(){
    //在这,我们分别预存储了第一个和最后一个可见位置之外的3个位置的bitmap
    //即dataCache中始终只缓存了(M=6+Gallery当前可见view的个数)M个bitmap
		int start = mGallery.getFirstVisiblePosition()-3;
		int end = mGallery.getLastVisiblePosition()+3;
		Log.v(TAG, "start:"+ start);
		Log.v(TAG, "end:"+ end);
		//释放position<start之外的bitmap资源
		Bitmap delBitmap;
		for(int del=0;del<start;del++){
			delBitmap = dateCache.get(del);
			if(delBitmap != null){
                                //如果非空则表示有缓存的bitmap,需要清理
				Log.v(TAG, "release position:"+ del);
				//从缓存中移除该del->bitmap的映射
                                dateCache.remove(del);
				delBitmap.recycle();
			}
		}

		freeBitmapFromIndex(end);
		
	}
	
	/**
	 * 从某一位置开始释放bitmap资源
	 * @param index
	 */
	private void freeBitmapFromIndex(int end) {
		//释放之外的bitmap资源
		Bitmap delBitmap;
		for(int del =end+1;del<dateCache.size();del++){
			delBitmap = dateCache.get(del);
			if(delBitmap != null){
				dateCache.remove(del);
				delBitmap.recycle();
				Log.v(TAG, "release position:"+ del);
			}
			
		}
	} 
经过这些额外的操作,有效的避免了OOM的问题。
分享到:
评论

相关推荐

    ANDROIDBITMAP内存限制OOM,OUTOFMEMORY.pdf

    文档标题和描述中提到的“ANDROIDBITMAP内存限制OOM,OUTOFMEMORY”指的就是在处理位图(BITMAP)时超出了虚拟机(VM)的内存预算,导致系统抛出OutOfMemoryError异常。 根据给出的内容部分,我们可以推断出以下知识...

    android bitmap outOfMemory解决方法

    6. **使用SoftReference或WeakReference**:使用这两种引用类型可以降低Bitmap的内存保留优先级,使系统在内存紧张时更容易回收Bitmap。 7. **监控内存使用**:利用Android的`ActivityManager`或第三方工具如MAT...

    bitmap内存问题

    内存溢出(Out Of Memory,简称OOM)通常发生在应用消耗的物理内存超过了系统分配给它的最大内存限制。对于Android而言,当一个进程使用的本机或Java堆内存超过了一定阈值时,就会触发OOM错误。`Bitmap`作为占用内存...

    Android避免内存溢出(Out of Memory)方法汇总

    在Android开发中,内存管理是至关重要的,尤其是避免内存溢出(Out of Memory,简称OOM)。内存溢出会导致应用程序崩溃,影响用户体验。本篇文章将详细阐述如何在Android中有效地防止内存溢出,主要包括理解不同类型...

    通过压缩Bitmap来避免OOM

    在Android开发中,Bitmap对象是消耗内存的主要因素之一,尤其在处理大图或者大量图片时,容易引发“Out of Memory”(简称OOM)错误。为了避免这种情况,开发者需要掌握如何有效地管理和压缩Bitmap,以优化内存使用...

    Android Bitmap 处理示例

    然而,由于Bitmap对象通常占用大量的内存,不当的处理可能导致内存溢出(Out Of Memory)问题,因此对Bitmap进行高效管理是至关重要的。本示例将详细介绍Android中Bitmap的处理,包括图片缓存策略和加载大图的技巧。...

    android 轻松避免内存溢出

    在Android开发中,内存管理是至关重要的,尤其是处理图片资源时,经常遇到内存溢出(Out Of Memory,简称OOM)的问题。本项目提供了一个在Android 1.6 SDK环境下编写的工具类,旨在帮助开发者有效地避免内存溢出,...

    Demo_BitmapOOM_Solution.rar

    Bitmap OOM(Out Of Memory)问题在Android开发中是一个常见的挑战,尤其是在处理大量图片或者高分辨率图片时。这个"Demo_BitmapOOM_Solution.rar"压缩包文件很可能是为了解决Bitmap内存溢出问题提供的一种解决方案...

    bitmap图片压缩

    Bitmap占用内存较大,如果处理不当,可能导致内存溢出(Out Of Memory,OOM)问题。因此,对Bitmap进行压缩至关重要。 1. **按比例压缩(Scale Compress)**: 这种方法通过改变Bitmap的宽度和高度来减小其大小。...

    压缩bitmap的较优方案

    在Android开发中,Bitmap对象是用于处理图像的主要类,但如果不妥善管理,它可能会消耗大量内存,引发“OutOfMemory”(OOM)错误。为了解决这个问题,开发者需要掌握一些优化Bitmap压缩的策略。以下是一些关于如何...

    Android内存管理

    **2.4 Android-避免出现bitmap内存限制OUTOFMEMORY的一种方法** Bitmap是Android中用于表示图像的重要类,但是不当使用容易导致内存溢出。为了避免这种情况,可以采取以下措施: - **减小Bitmap尺寸**: 通过压缩...

    android文本/图片切换,contentresolver,抽屉,评级,scroll,bitmap使用

    但Bitmap消耗内存较大,不当使用可能导致内存泄漏或OOM(Out Of Memory)异常。因此,我们需要合理使用`BitmapFactory.Options`来控制解码时的大小,使用`Bitmap.createScaledBitmap()`缩放图像,以及使用`Recycle()...

    Android 内存溢出问题

    然而,如果应用程序在短时间内分配大量内存,或者持有大量无法释放的对象引用,就可能导致内存溢出(Out Of Memory,OOM)。 例如,错误信息"02-04 21:46:08.703: ERROR/dalvikvm-heap(2429): 1920000-byte ...

    android双缓冲解决图片内存溢出的问题

    在Android开发中,处理大量的图片资源时,常常会遇到内存溢出(Out Of Memory,简称OOM)的问题。这是因为Android系统为每个应用分配的内存有限,当加载过多或过大的图片时,很容易超出这个限制,导致应用崩溃。为了...

    避免bitmapoomTestAsyncImageLoaderImageView

    在Android开发中,Bitmap对象是图像处理的核心,但如果不妥善管理,可能会引发“Bitmap Out Of Memory”(简称BitmapOOM)错误。BitmapOOM是由于Android系统内存限制,加载大图或大量图片时耗尽内存导致的。针对这个...

    安卓Android源码——防止内存溢出浅析.zip

    Android系统使用Java虚拟机(Dalvik或ART)进行内存分配和管理,但如果不妥善处理,很容易引发内存溢出(Out Of Memory,简称OOM)。本篇文章将深入浅析Android源码中的内存管理机制,以及如何预防和处理内存溢出...

    Android应用源码之防止内存溢出浅析.zip

    在Android应用开发中,内存管理是一项至关重要的任务,因为不当的内存使用可能导致应用程序崩溃或性能下降,最常见的问题就是内存溢出(Out Of Memory,简称OOM)。本资源"Android应用源码之防止内存溢出浅析"着重...

    安卓图片压缩类,避免内存溢出OOM

    在安卓开发中,图片处理是一项常见且重要的任务,然而,如果不妥善处理,它可能会导致一个严重的问题——内存溢出(Out Of Memory,简称OOM)。内存溢出是由于程序请求的内存超过了系统分配的最大内存,从而导致程序...

    安卓Android源码——图片下载以及内存处理防OOM.rar

    在安卓开发中,图片加载和内存管理是两个关键的领域,尤其对于防止应用程序出现内存溢出(Out Of Memory,简称OOM)至关重要。Android系统运行在有限的内存资源中,因此,开发者需要谨慎处理图片资源,避免过度消耗...

    防止内存溢出浅析

    内存溢出(Out of Memory,简称OOM)是计算机科学中的一种常见问题,特别是在资源有限的移动设备如Android手机上,处理不当可能导致应用崩溃。本文将深入浅谈如何防止Android应用程序中的内存溢出。 首先,我们需要...

Global site tag (gtag.js) - Google Analytics