图片oom问题

Posted znsongshu

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了图片oom问题相关的知识,希望对你有一定的参考价值。

1、什么是OOM?
程序申请内存过大,虚拟机无法满足我们,然后自杀了。这个现象通常出现在大图片的APP开发,或者需要用到很多图片的时候。通俗来讲就是我们的APP需要申请一块内存来存放图片的时候,系统认为我们的程序需要的内存过大,及时系统有充分的内存,比如1G,但是系统也不会分配给我们的APP,故而抛出OOM异常,程序没有捕捉异常,故而弹窗崩溃了
2、为什么会有OOM?
因为android系统的APP每个进程或者虚拟机有最大内存限制,一旦超过这个限制系统就会抛出OOM错误。跟手机剩余内存是否充足没有多少关系。
3、为什么Android会有APP的内存限制
(1)要开发者使用内存更加合理。限制每个应用可用内存上限,避免恶意程序或单个程序使用过多内存导致其他程序的不可运行。有了限制,开发者就必须合理使用资源,优化资源使用
(2)屏幕显示内容有限,内存足够即可。即使有万千图片千万数据需要使用到,但在特定时刻需要展示给用户看的总是有限的,因为屏幕显示就那么大,上面可以放的信息就是很有限的。大部分信息都是处于准备显示状态,所以没必要给予太多heap内存。必须一个ListView显示图片,打个比方这个ListView含有500个item,但是屏幕显示最多有10调item显示,其余数据是处于准备显示状态。
(3)Android多个虚拟机Davlik的限制需要。android设备上的APP运行,每打开一个应用就会打开至少一个独立虚拟机。这样可以避免系统崩溃,但代价是浪费更多内存。
4、有GC自动回收资源,为什么还会有OOM?
Android的GC会按照特定的算法来回收不使用的资源,但是gc一般回收的是无主的对象内存或者软引用资源。
使用软引用的图片资源在一定程度上可以避免OOM。
ps:不用的对象设置为null,是一个好习惯。不过更好的方法是,不用的图片直接recycle。因为有时候通过设置null让gc来回收还是来不及。
5、怎么来避免OOM产生呢?
简单通过SoftReference引用方式管理图片资源
建一个SoftReference的hashmap,使用图片时,先检查这个hashmap是否有softreference,softreference的图片是否为空,如果为空将图片加载到softreference并加入haspmap。
代码如下:

import java.io.InputStream;  
import java.lang.ref.SoftReference;  
import java.net.HttpURLConnection;  
import java.net.URL;  
import java.net.URLConnection;  
import java.util.HashMap;  
import android.content.Context;  
import android.graphics.Bitmap;  
import android.graphics.BitmapFactory;  
import android.os.Handler;  
import android.os.Handler.Callback;  
import android.os.Message;  
import android.widget.ImageView;  
  
  
/** 
 * 功能说明:异步加载图片 
 * 
 */  
  
public class AsyncImageLoaderCore {  
    public Context context; // 做本地缓存时会用到  
    public HashMap<String, SoftReference<Bitmap>> imageCache;// 软引用集合  
     
    public AsyncImageLoaderCore(Context context) {  
        this.context = context;  
        this.imageCache = new HashMap<String, SoftReference<Bitmap>>();  
    }  
  
  
    public Bitmap loadBitmap(final String imageUrl, final ImageView imageView, final ImageCallback imageCallback) {  
        if (imageCache.containsKey(imageUrl)) {  
            SoftReference<Bitmap> softReference = imageCache.get(imageUrl);  
            if (softReference.get() != null)  
                return softReference.get();  
        }  
          
        final Handler handler = new Handler(new Callback() {  
            @Override  
            public boolean handleMessage(Message msg) {  
                imageCallback.imageLoaded((Bitmap) msg.obj, imageView, imageUrl);  
                return false;  
            }  
        });  
          
        new Thread() {  
            @Override  
            public void run() {  
                Bitmap bitmap = null;  
                try {  
                    bitmap = getHttpBitmap(imageUrl);  
                } catch (Exception e) {  
                    e.printStackTrace();  
                    return;  
                }  
                  
                if (null != bitmap) {  
                    imageCache.put(imageUrl, new SoftReference<Bitmap>(bitmap));  
                    handler.sendMessage(handler.obtainMessage(0, bitmap));  
                }  
            }  
        }.start();  
        return null;  
    }  
    private final int MAX_PIC_LENGTH = 200000;// 最大字节长度限制[可调,最好不要超过200000]  
    private final int SAMPLE_SIZE = 14;// 裁剪图片比列(1/14)[可调]  
    /** 
     * 获取网络图片 
     */  
    private Bitmap getHttpBitmap(String imgUrl) throws Exception {  
        URL htmlUrl = new URL(imgUrl);  
        URLConnection connection = htmlUrl.openConnection();  
        HttpURLConnection conn = (HttpURLConnection) connection;  
        if (conn.getResponseCode() == HttpURLConnection.HTTP_OK) {  
            InputStream inputStream = conn.getInputStream();  
            byte[] bytes = changeToBytes(inputStream);  
            if (bytes.length < MAX_PIC_LENGTH) {  
                return BitmapFactory.decodeByteArray(bytes, 0, bytes.length);  
            } else if (bytes.length < MAX_PIC_LENGTH * SAMPLE_SIZE) {  
                BitmapFactory.Options options = new BitmapFactory.Options();  
                options.inJustDecodeBounds = false;  
                options.inSampleSize = SAMPLE_SIZE;  
                return BitmapFactory.decodeByteArray(bytes, 0, bytes.length, options);  
            }  
        }  
        return null;  
    }  
  
  
    /** 
     * 将流转换成字节数组 
     */  
  
    public byte[] changeToBytes(InputStream inputStream) throws Exception  
    {  
        ByteArrayOutputStream outputStream = new ByteArrayOutputStream();  
        byte[] buffer = new byte[1024];// 每次读取的字节长度  
        int len = 0;  
        while ((len = inputStream.read(buffer)) != -1)  
        {  
            outputStream.write(buffer, 0, len);  
        }  
        inputStream.close();  
        return outputStream.toByteArray();  
    }  
    /** 
     * 异步加载资源回调接口 
     */  
    public interface ImageCallback {  
        public void imageLoaded(Bitmap bitmap, ImageView imageView, String imageUrl);  
    }  
}  ```

 


以上是关于图片oom问题的主要内容,如果未能解决你的问题,请参考以下文章

Android OOM 排查与解决——图片加载优化

如何解决上传多张图片时遇到的oom问题

android图片压缩避免OOM

Android BitmapFactory.Options 解决大图片加载OOM问题

viewPager--viewpager时,发生内存溢出OOM问题

图片过大导致OOM