原地址:http://zhanhao.iteye.com/blog/1463350
android手机给应用分配的内存通常是8兆左右,如果处理内存处理不当很容易造成OutOfMemoryError,我们的产品出现最多的错误也是OutOfMemoryError的异常,
在解决这个异常时在网上发现很多关于OutOfMemoryError的原因的介绍。
OutOfMemoryError主要由以下几种情况造成:
1.数据库的cursor没有关闭。
操作Sqlite数据库时,Cursor是数据库表中每一行的集合,Cursor提供了很多方法,可以很方便的读取数据库中的值,
可以根据索引,列名等获取数据库中的值,通过游标的方式可以调用moveToNext()移到下一行
当我们操作完数据库后,一定要记得调用Cursor对象的close()来关闭游标,释放资源。
2.构造adapter没有使用缓存contentview。
在继承BaseAdapter时会让我们重写getView(int position, View convertView, ViewGroup parent)方法,
第二个参数convertView就是我们要用到的重用的对象
- @Override
- public View getView(int position, View convertView, ViewGroup parent) {
- ViewHolder vHolder = null;
- //如果convertView对象为空则创建新对象,不为空则复用
- if (convertView == null) {
- convertView = inflater.inflate(..., null);
- // 创建 ViewHodler 对象
- vHolder = new ViewHolder();
- vHolder.img= (ImageView) convertView.findViewById(...);
- vHolder.tv= (TextView) convertView
- .findViewById(...);
- // 将ViewHodler保存到Tag中
- convertView.setTag(vHolder);
- } else {
- //当convertView不为空时,通过getTag()得到View
- vHolder = (ViewHolder) convertView.getTag();
- }
- // 给对象赋值,修改显示的值
- vHolder.img.setImageBitmap(...);
- vHolder.tv.setText(...);
- return convertView;
- }
- //将显示的View 包装成类
- static class ViewHolder {
- TextView tv;
- ImageView img;
- }
这里只讲使用方法,具体性能测试文章请见:
ListView中getView的原理+如何在ListView中放置多个item
http://www.cnblogs.com/xiaowenji/archive/2010/12/08/1900579.html
Android开发之ListView适配器(Adapter)优化
http://shinfocom.iteye.com/blog/1231511
3.调用registerReceiver()后未调用unregisterReceiver().
广播接收者(BroadcastReceiver)经常在应用中用到,可以在多线程任务完成后发送广播通知UI更新,也可以接收系统广播实现一些功能
可以通过代码的方式注册:
IntentFilter postFilter = new IntentFilter();
postFilter.addAction(getPackageName() + ".background.job");
this.registerReceiver(receiver, postFilter);
当我们Activity中使用了registerReceiver()方法注册了BroadcastReceiver,一定要在Activity的生命周期内调用unregisterReceiver()方法取消注册
也就是说registerReceiver()和unregisterReceiver()方法一定要成对出现,通常我们可以重写Activity的onDestory()方法:
- @Override
- protected void onDestroy() {
- this.unregisterReceiver(receiver);
- super.onDestroy();
- }
4.未关闭InputStream/OutputStream。
这个就不多说了,我们操作完输入输出流都要关闭流
5.Bitmap使用后未调用recycle()。
图片处理不好是造成内存溢出的又一个头号原因,(在我们的产品中也有体现),
当我们处理完图片之后可以通过调用recycle()方法来回收图片对象
- if(!bitmap.isRecycled())
- {
- bitmap.recycle()
- }
除此之外:
直接使用ImageView显示bitmap会占用较多资源,特别是图片较大的时候,可能导致崩溃。
使用BitmapFactory.Options设置inSampleSize, 这样做可以减少对系统资源的要求。
属性值inSampleSize表示缩略图大小为原始图片大小的几分之一,即如果这个值为2,则取出的缩略图的宽和高都是原始图片的1/2,图片大小就为原始大小的1/4。
BitmapFactory.Options bitmapFactoryOptions = new BitmapFactory.Options();
bitmapFactoryOptions.inJustDecodeBounds = true;
bitmapFactoryOptions.inSampleSize = 2;
// 这里一定要将其设置回false,因为之前我们将其设置成了true
// 设置inJustDecodeBounds为true后,decodeFile并不分配空间,即,BitmapFactory解码出来的Bitmap为Null,但可计算出原始图片的长度和宽度
options.inJustDecodeBounds = false;
Bitmap bmp = BitmapFactory.decodeFile(sourceBitmap, options);
6.Context泄漏。
这是一个很隐晦的OutOfMemoryError的情况。先看一个Android官网提供的例子:
- private static Drawable sBackground;
- @Override
- protected void onCreate(Bundle state) {
- super.onCreate(state);
- TextView label = new TextView(this);
- label.setText("Leaks are bad");
- if (sBackground == null) {
- sBackground = getDrawable(R.drawable.large_bitmap);
- }
- label.setBackgroundDrawable(sBackground);
- setContentView(label);
- }
这段代码效率很快,但同时又是极其错误的;
在第一次屏幕方向切换时它泄露了一开始创建的Activity。当一个Drawable附加到一个 View上时,
View会将其作为一个callback设定到Drawable上。上述的代码片段,意味着Drawable拥有一个TextView的引用,
而TextView又拥有Activity(Context类型)的引用,换句话说,Drawable拥有了更多的对象引用。即使Activity被 销毁,内存仍然不会被释放。
另外,对Context的引用超过它本身的生命周期,也会导致Context泄漏。所以尽量使用Application这种Context类型。
这种Context拥有和应用程序一样长的生命周期,并且不依赖Activity的生命周期。如果你打算保存一个长时间的对象,
并且其需要一个 Context,记得使用Application对象。你可以通过调用Context.getApplicationContext()或 Activity.getApplication()轻松得到Application对象。
最近遇到一种情况引起了Context泄漏,就是在Activity销毁时,里面有其他线程没有停。
总结一下避免Context泄漏应该注意的问题:
1.使用Application这种Context类型。
2.注意对Context的引用不要超过它本身的生命周期。
3.慎重的使用“static”关键字。
4.Context里如果有线程,一定要在onDestroy()里及时停掉。
7.static关键字
当类的成员变量声明成static后,它是属于类的而不是属于对象的,如果我们将很大的资源对象(Bitmap,context等)声明成static,那么这些资源不会随着对象的回收而回收,
会一直存在,所以在使用static关键字定义成员变量的时候要慎重。
相关推荐
"Android 内存泄露 Mat工具分析" Android 内存泄露是一个常见的问题,它会导致应用程序的性能下降,甚至崩溃。Memory Analyzer Tool(MAT)是一个功能强大且广泛使用的内存泄露分析工具。本文将详细介绍如何使用 ...
### Android内存泄露测试详解 在Android开发中,内存管理是一项至关重要的任务,不当的内存管理会导致应用运行缓慢、崩溃甚至耗尽系统资源。内存泄露是内存管理中的一个常见问题,它指的是应用程序分配了内存但未能...
总结,Android内存优化是一个系统性的工作,涉及到多个层面,开发者需要深入理解内存管理机制,熟练掌握各种优化技巧,才能打造出运行流畅、占用资源合理的应用。通过持续学习和实践,开发者可以不断提升自己在内存...
总结来说,Android内存优化的核心在于避免内存泄露和合理分配内存资源。通过理解内存泄露的原因,运用恰当的设计模式和编程技巧,同时借助工具来监控和分析内存使用情况,可以有效地提升应用的性能,延长应用的生命...
### Android性能优化之内存泄露 #### 一、内存泄露概述 在Android开发中,内存管理是确保应用程序稳定性和性能的关键因素之一。不当的内存管理可能导致内存泄露,进而引发一系列问题,如应用程序运行缓慢、崩溃...
Android内存优化——常见内存泄露及优化方案 对于安卓优化见解独到,一针见血,从此再无此问题
常见的Android内存泄露类型包括: 1. **静态变量引用**:当一个静态变量引用了一个Activity或Context,这个对象就无法被GC回收,因为静态变量在整个应用生命周期内都存在。解决方案是避免使用静态变量持有Activity...
关于android内存泄露的性能优化的PPT说明,解释以及建议解决的办法
以下将详细介绍如何使用MAT进行Android内存泄露的检测。 **安装MAT** MAT可以作为独立的应用程序运行,也可以作为Eclipse IDE的插件。下载MAT的相应版本(例如MemoryAnalyzer-1.1.1.20110824-linux.gtk.x86_64.zip...
总结来说,Android应用内存泄露是一个复杂的问题,涉及到Android内存管理机制、应用设计和代码实现等多个方面。通过上述方法,我们可以有效地检测、定位和规避内存泄露问题,从而提升Android应用的性能和稳定性。
在Android开发中,ListView是展示大量数据常用的组件,但如果不进行优化,很容易引发内存溢出和界面卡顿问题,特别是当列表中包含大图时。本文将深入探讨如何对ListView进行内存优化,确保10M级别的图片加载时,应用...
本文将深入探讨Android App中的内存泄露问题,以及如何定位和规避这些内存泄露,确保应用运行流畅。 首先,我们需要理解什么是内存泄露。在计算机科学中,内存泄露是指程序在申请内存后,无法释放已申请的内存空间...
在Android开发过程中,内存泄露是一个严重的问题,它可能导致应用性能下降、用户界面卡顿甚至应用崩溃。LeakCanary是一款由Square公司开发的强大的内存泄漏检测工具,特别为Android平台设计,可以帮助开发者轻松定位...
总结来说,Android内存优化主要包括理解并控制内存分配、避免内存泄漏、谨慎使用static和线程。开发者应关注对象的生命周期,确保及时释放不再使用的资源,并合理设计数据结构和引用关系,以降低内存压力。同时,...
这个名为"Android学习资料之内存优化.zip"的压缩包包含了一些关于Android内存优化的重要文档,这些文档深入探讨了常见的内存泄露问题以及对应的优化策略。以下是根据这些知识点进行的详细解读: 1. **内存泄露**:...
总的来说,HeapSnap作为一款专门的Android内存泄露检测工具,通过其丰富的功能和直观的报告,使得开发者能高效地定位和解决内存泄露问题,从而提升应用的性能和稳定性。结合Android内存管理的基本原理和最佳实践,...
3. **LeakCanary**:专为Android开发的内存泄露检测库,可自动检测并报告内存泄漏。 4. **Node.js Inspector**:用于Node.js应用程序的内存分析,可帮助识别内存泄漏。 5. **Valgrind**:针对C/C++应用程序的工具...
本文将深入探讨Android内存泄漏的概念、常见场景、预防措施以及检测方法。 首先,Android虚拟机的垃圾回收(GC)采用的是根搜索算法。这个算法从GC Roots(包括全局静态变量、活动线程、JNI引用等)开始遍历堆中的...