今天做项目,发现需要显示一张超大图片,处理过后,还有561Kb
加载的时候,就crash --- OOM
shortMsg:java.lang.OutOfMemoryError
longMsg:java.lang.OutOfMemoryError: bitmap size exceeds VM budget
stackTrace:java.lang.OutOfMemoryError: bitmap size exceeds VM budget
at android.graphics.Bitmap.nativeCreate(Native Method)
at android.graphics.Bitmap.createBitmap(Bitmap.java:477)
at android.graphics.Bitmap.createBitmap(Bitmap.java:444)
at android.graphics.Bitmap.createScaledBitmap(Bitmap.java:349)
at android.graphics.BitmapFactory.finishDecode(BitmapFactory.java:512)
at android.graphics.BitmapFactory.decodeStream(BitmapFactory.java:487)
at android.graphics.BitmapFactory.decodeResourceStream(BitmapFactory.java:336)
代码如下:
detailView=(ImageView)findViewById(R.id.detailView);
detailView.setBackgroundResource(R.drawable.more_info);//this line will lead to OOM
换成这种:
detailView.setImageResource(R.drawable.more_info); //也同样会OOM
后来找到了solution:
/**
* 以最省内存的方式读取本地资源的图片
* @param context
*@param resId
* @return
*/
public static Bitmap readBitMap(Context context, int resId){
BitmapFactory.Options opt = new BitmapFactory.Options();
opt.inPreferredConfig = Bitmap.Config.RGB_565;
opt.inPurgeable = true;
opt.inInputShareable = true;
//获取资源图片
InputStream is = context.getResources().openRawResource(resId);
return BitmapFactory.decodeStream(is,null,opt);
}
取得bitmap之后,再 detailView.setImageBitmap(pdfImage); 就ok了!
那是为什么,会导致oom呢:
原来当使用像 imageView.setBackgroundResource,imageView.setImageResource, 或者 BitmapFactory.decodeResource 这样的方法来设置一张大图片的时候,
这些函数在完成decode后,最终都是通过java层的createBitmap来完成的,需要消耗更多内存。
因此,改用先通过BitmapFactory.decodeStream方法,创建出一个bitmap,再将其设为ImageView的 source,decodeStream最大的秘密在于其直接调用JNI>>nativeDecodeAsset()来完成decode,无需再使用java层的createBitmap,从而节省了java层的空间。如果在读取时加上图片的Config参数,可以跟有效减少加载的内存,从而跟有效阻止抛out of Memory异常。
另外,需要特别注意:
decodeStream是直接读取图片资料的字节码了, 不会根据机器的各种分辨率来自动适应,使用了decodeStream之后,需要在hdpi和mdpi,ldpi中配置相应的图片资源,否则在不同分辨率机器上都是同样大小(像素点数量),显示出来的大小就不对了。
相关推荐
在Android开发中,图片的加载和动画效果是一个重要的部分,特别是在UI交互中。这个主题“Android用线程实现ImageView图片变换+可以停止和继续”主要关注如何利用线程技术来实现图片的动态变换,并且提供了暂停和继续...
在Android开发中,由于内存限制,处理大量图片时很容易引发OutOfMemoryError(简称OOM),导致应用崩溃。为了解决这个问题,开发者通常会采用图片缓存技术,其中三级缓存是一种常用的策略。本文将深入探讨如何利用三...
在Android开发中,高效加载大图和多图是至关重要的,因为如果不加以处理,这可能导致程序出现内存溢出(OOM)异常,严重影响用户体验。本文将详细介绍如何有效地避免此类问题。 首先,我们需要理解Android应用程序...
在Android开发中,处理大容量的GIF图片可能会导致内存溢出(OOM)问题,这是因为Android系统默认使用Java层的Bitmap对象来加载和显示图片,而这种操作会消耗大量内存。针对这一问题,一种解决方案是利用原生代码(C/...
在 Android 应用程序的设计中,几乎不可避免地都需要加载和显示图片,由于不同的图片在大小上千差万别,有些图片可能只需要几十KB的内存空间,有些图片却需要占用几十MB的内存空间;或者一张图片不需要占用太多的...
在Android开发中,图片压缩是一项重要的技术,尤其在移动设备资源有限的情况下,处理好图片的大小和质量至关重要。本文将详细讲解Android中的图片压缩技术,包括为什么要进行图片压缩、常用的图片压缩方法以及如何...
在Android开发中,加载大图片是一项需要注意的重要任务,因为不恰当的处理可能导致应用程序出现内存溢出(Out Of Memory,OOM)错误。以下是Android加载大图片的一些关键知识点和实践方法: 1. **Bitmap对象的使用*...
在Android开发中,图片压缩是一项重要的技术,尤其在处理大图片和多图片场景时,能够有效地防止因内存消耗过大导致的Out Of Memory (OOM)异常。本文将详细讲解两种常用的Android图片压缩方法,并提供源码供参考。 1...
在Android开发中,加载大图和多图是一项挑战,因为如果不妥善处理,很容易引发内存溢出(Out Of Memory,简称OOM)异常。Android系统为每个应用程序分配了一定的内存限制,当应用程序占用的内存超过这个限制时,系统...
在Android开发中,图片压缩是一项重要的技术,尤其考虑到设备资源有限和用户体验的需求。本文将深入探讨Android图片压缩的原理和实现方法,以及如何避免因图片过大导致的内存溢出(Out Of Memory,简称OOM)问题。 ...