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); } } ```
原文地址:https://www.cnblogs.com/znsongshu/p/9379900.html
时间: 2024-11-10 13:29:48