Android 内存溢出管理与测试
今天发现正在做的项目,时不时的会报错:dalvikvm heap out of memory on a 7458832-byte allocation (堆分配的内存溢出)
为什么会内存溢出呢?我以前从未遇见这种情况。后来在网上查了查资料,还是挺多的。
怎么说呢?因为Android开发基本上是以java语言为基础,那么程序是在java虚拟机上运行的。而虚拟机不允许单个程序中的Bitmap占用超过8M的内存,从报错的日志可以看出:7458832-byte大约就是7M多的样子,基本吻合上述数据。在我的项目中大量的使用到了各种图片,因此是因为图片导致内存溢出的的可能性很高。
那么,下面就提供一下解决方案:
第一种方案:
Android堆内存也可自己定义大小 和 优化Dalvik虚拟机的堆内存分配(LZ没试过,不过网上很多人都说可行)。
强制定义自己软件的对内存大小,我们使用Dalvik提供的 dalvik.system.VMRuntime类来设置最小堆内存为例:
private final static int CWJ_HEAP_SIZE = 6* 1024* 1024 ;
VMRuntime.getRuntime().setMinimumHeapSize(CWJ_HEAP_SIZE); //设置最小heap内存为6MB大小。当然对于内存吃紧来说还可以通过手动干涉GC去处理
第二种方案:
手动的回收内存。
示例代码如下:bit为Bitmap对象
if (bit != null && !bit.isRecycled()) {
bit.recycle();
bit = null;
}
System.gc();
bitmap.recycle()方法用于回收该bitmap所占用的内存,用System.gc()调用一下系统的垃圾回收器。
需要注意的是:回收内存要及时,比如说SurfaceView,就应该在onSurfaceDestroyed这个方法中回收。如果Activity使用了bitmap,就可以在onStop或者onDestroy方法中回收等等。
我首先就是用的这种方法,修改后发现bug出现的频率减小了,但仍然存在。
第三种方案:
既然是图的问题,那就从图片下手。就是使图片体积大小变小。
这也分为两个方面:
1、分辨率不变,图片大小减小2、分辨率改变,图片减小。(用PS都很容易的)
需要注意的是:不要减小得太小而影响了人眼看上去的美感。
以上三种方案,后两种较简单且常用。
另外,因为是Android的图片导致的内存问题,那么就总结了一下与图片相关的方法的性能,结果很有收获啊……(不看不知道,一看吓一跳)
测试的环境为:笔记本i3处理器,64位win7系统,测试手机为小米2S,测试图片大小为500KB
比较Drawable与Bitmap占用内存大小
比较BitmapFactory类的decodeResource方法与decodeStream方法的效率
以及优化后的BitmapFactory类的decodeResource方法。
下面贴上代码与运行图:(每次请只运行一个方法,隐藏另外的三种方法)
public class MainActivity extends Activity {
int number = 1000;
Drawable[] array;
Bitmap bitmap[];
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
testDrawable();
// testBitmap_decodeResource();
// testBitmap_decodeStream();
//testnewBitmap_decodeResource();
}
/**
* 优化后的BitmapFactory类的decodeResource方法
*/
private void testnewBitmap_decodeResource() {
bitmap = new Bitmap[number];
for (int i = 0; i < number; i++) {
Log.i("", "测试第" + (i + 1) + "张图片");
//压缩,用于节省BITMAP内存空间--解决BUG的关键步骤
BitmapFactory.Options opts = new BitmapFactory.Options();
opts.inSampleSize = 2;//这个的值压缩的倍数(2的整数倍),数值越小,压缩率越小,图片越清晰
//返回原图解码之后的bitmap对象
bitmap[i] = BitmapFactory.decodeResource(getResources(),
R.drawable.begin_background, opts);
}
}
/**
* BitmapFactory类的decodeStream方法
*/
private void testBitmap_decodeStream() {
bitmap = new Bitmap[number];
for (int i = 0; i < number; i++) {
Log.i("", "测试第" + (i + 1) + "张图片");
bitmap[i] = BitmapFactory.decodeStream(getResources()
.openRawResource(R.drawable.begin_background));// 这里换了方法
}
}
/**
* BitmapFactory类的decodeResource方法
*/
private void testBitmap_decodeResource() {
bitmap = new Bitmap[number];
for (int i = 0; i < number; i++) {
Log.d("", "测试第" + (i + 1) + "张图片");
bitmap[i] = BitmapFactory.decodeResource(getResources(),
R.drawable.begin_background);
}
}
/**
* Drawable的使用
*/
private void testDrawable() {
array = new BitmapDrawable[number];
for (int i = 0; i < number; i++) {
Log.w("", "测试第" + (i + 1) + "张图片");
array[i] = getResources().getDrawable(R.drawable.begin_background);
}
}
}
首先贴出 Drawable的使用 的运行结果:(顺利测试完1000张图片)

BitmapFactory类的decodeResource方法运行结果:(居然只有11张)

运行优化后的BitmapFactory类的decodeResource方法:(优化后可达到46张,且opts.inSampleSize = n;N的参数设置得越大可达到的数量越多,但损失了图片质量)

BitmapFactory类的decodeStream方法:(可达22张图片)

这下,相信大家对Android中与图片相关的这几种方法的性能有直观的认识了吧。其实,在测试时除了看到数量外,还可以看到运行时间上的差距。
其实,综合上面产生bitmap方法,可以得出一种最优的方法:(环境以上面其他方法相同)
public void test() {
bitmap = new Bitmap[number];
for (int i = 0; i < number; i++) {
Log.i("", "测试第" + (i + 1) + "张图片");
BitmapFactory.Options options = new BitmapFactory.Options();
options.inSampleSize = 2;
bitmap[i] = BitmapFactory.decodeStream(getResources()
.openRawResource(R.drawable.begin_background), null,
options);
}
}
最终测试得到的数据是,可以测试80多张图片。(图片也与其他方法测试的相同)
Android 内存溢出管理与测试的更多相关文章
- 【转载】Android 内存溢出如何发生的。
[转载]Android 内存溢出如何发生的. 且谈Android内存溢出 前言 关于android的内存溢出在创新文档库中也有不少,网络上也有很多这方面的资料.所以这遍文章不算是正真意义上的创新,仅仅 ...
- Android 内存溢出处理方案
转自 : http://www.cnblogs.com/hello-ruby/archive/2013/04/19/3031098.html 首先我们来看看android内存溢出的原因,有可能是: 由 ...
- android 内存溢出问题分析
最近的项目中,内存一直再增长,但是不知道是什么问题,导致内存溢出,在网上看到了这么一篇关于内存分析与管理的文章,解决了部分问题,感觉这篇文 章还不错,就转帖到我的blog上了,希望对大家有所帮助. ...
- Android内存溢出、内存泄漏常见案例及最佳实践总结
内存溢出是Android开发中一个老大难的问题,相关的知识点比较繁杂,绝大部分的开发者都零零星星知道一些,但难以全面.本篇文档会尽量从广度和深度两个方面进行整理,帮助大家梳理这方面的知识点(基于Jav ...
- Android内存溢出解决方案(OOM)
众所周知,每个Android应用程序在运行时都有一定的内存限制,限制大小一般为16MB或24MB(视平台而定).因此在开发应用时需要特别关注自身的内存使用量,而一般最耗内存量的资源,一般是图片.音频文 ...
- Android 内存溢出解决方案(OOM) 整理总结
在最近做的工程中发现加载的图片太多或图片过大时经常出现OOM问题,找网上资料也提供了很多方法,但自己感觉有点乱,特此,今天在不同型号的三款安卓手机上做了测试,因为有效果也有结果,今天小马就做个详细的总 ...
- Android 内存溢出解决方案(OOM) 整理总结<转>
在最近做的工程中发现加载的图片太多或图片过大时经常出现OOM问题,找网上资料也提供了很多方法,但自己感觉有点乱,特此,今天在不同型号的三款安卓手机上做了测试,因为有效果也有结果,今天小马就做个详细的总 ...
- Android内存进程管理机制
参考文章: http://www.apkbus.com/android-104940-1-1.htmlhttp://blog.sina.com.cn/s/blog_3e3fcadd0100yjo2.h ...
- android 内存溢出与内存泄露
内存溢出就是软件运行需要的内存,超出了java虚拟机给他分配的可用的最大内存 内存泄露就是在缓存图片文字等等的时候,没有关闭流所导致的内存泄露
随机推荐
- mac linux netstat -n
一直都不明白为什么mac就是不现实8080端口,现在明白了 USERtekiiMac-3:~ user$ netstat -an | grep tcp46 tcp46 0 0 *.8009 *.* L ...
- 一步一步学android控件(之十五) —— DegitalClock & AnalogClock
原本计划DigitalClock和AnalogClock单独各一篇来写,但是想想,两个控件的作用都一样,就和在一起写一篇了. DegitalClock和AnalogClock控件主要用于显示当前时间信 ...
- PowerDesigner 15 概述
PowerDesigner 15 概述 数据结构数据库powerbuildersybasemicrosoftuml 目录(?)[+] 一. PowerDesigner 介绍 PowerDesi ...
- HTML5要点(二)
<p> <b>今天周一</b>,<span>呵呵呵呵呵...</span> </p> <i>斜体文字</i&g ...
- linux系统基础(二)
磁盘管理(一) Linux设备认识 /dev/cdrom /dev/sr0 /dev/mouse /dev/sda /dev/hda IDE硬盘(支持4块):hd(a-d) [非IDE硬盘]SCSI硬 ...
- java常见面试题
JAVA相关基础知识 1.面向对象的特征有哪些方面 1.抽象: 抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面.抽象并不打算了解全部问题,而只是选择其中的一部分 ...
- Ubuntu 安装 Courier New字体
apt-get install ttf-mscorefonts-installer 它的本质是安装 Courier New字体 安装的时候会出现一个协议 按TAB键 ,可以选中<确定>按 ...
- SCOM MP中的PowerShell脚本
该脚本可在MP中运行 #param ($file,$threshold) #此处可定义file 和 threshold 参数,该参数可以从脚本外部接收 $threshold = 10 $file = ...
- C 双向链表
单链表的结点都只有一个指向下一个结点的指针 单链表的数据元素无法直接访问其前驱元素 逆序访问单链表中的元素是极其耗时的操作! len = LinkList_Length(list); for (i=l ...
- maven系列之二maven项目的创建和maven项目的结构
maven系列之一简单介绍了maven的基本信息,安装和配置,大家对maven有一个大概的了解,但是在maven项目开发中远远不够,为了进一步了解maven,现在我们介绍maven项目的创建和mave ...