Android中的内存管理机制以及正确的使用方式

概述

从操作系统的角度来说,内存就是一块数据存储区域,属于可被操作系统调度的资源。现代多任务(进程)的操作系统中,内存管理尤为重要,操作系统需要为每一个进程合理的分配内存资源,所以可以从两方面来理解操作系统的内存管理机制。

第一:分配机制。为每一个进程分配一个合理的内存大小,保证每一个进程能够正常的运行,不至于内存不够使用或者每个进程占用太多的内存。

第二:回收机制。在系统内存不足打的时候,需要有一个合理的回收再分配的机制,以保证新的进程可以正常运行。回收的时候就要杀死那些正在占有内存的进程,操作系统需要提供一个合理的杀死这些进程的机制,以保证更少的副作用。

而作为一个多进程的操作系统,Android系统对内存的管理,也是有一套自己的方法的。跟PC不一样的是,Android作为一个移动操作系统,一般情况下,内存资源会比PC更少,所以就需要更加谨慎的管理内存。

Android中的内存管理机制

分配机制

Android为每个进程分配内存的时候,采用了弹性的分配方式,也就是刚开始并不会一下分配很多内存给每个进程,而是给每一个进程分配一个“够用”的量。这个量是根据每一个设备实际的物理内存大小来决定的。随着应用的运行,可能会发现当前的内存可能不够使用了,这时候Android又会为每个进程分配一些额外的内存大小。但是这些额外的大小并不是随意的,也是有限度的,系统不可能为每一个App分配无限大小的内除。

Android系统的宗旨是最大限度的让更多的进程存活在内存中,因为这样的话,下一次用户再启动应用,不需要重新创建进程,只需要恢复已有的进程就可以了,减少了应用的启动时间,提高了用户体验。

回收机制

Android对内存的使用方式是“尽最大限度的使用”,这一点继承了Linux的优点。Android会在内存中保存尽可能多的数据,即使有些进程不再使用了,但是它的数据还被存储在内存中,所以Android现在不推荐显式的“退出”应用。因为这样,当用户下次再启动应用的时候,只需要恢复当前进程就可以了,不需要重新创建进程,这样就可以减少应用的启动时间。只有当Android系统发现内存不够使用,需要回收内存的时候,Android系统就会需要杀死其他进程,来回收足够的内存。但是Android也不是随便杀死一个进程,比如说一个正在与用户交互的进程,这种后果是可怕的。所以Android会有限清理那些已经不再使用的进程,以保证最小的副作用。

Android杀死进程有两个参考条件:

进程优先级:

Android为每一个进程分配了优先级的概念,优先级越低的进程,被杀死的概率就更大。Android中总共有5个进程优先级。具体含义这里不再给出。

前台进程:正常不会被杀死

可见进程:正常不会被杀死

服务进程:正常不会被杀死

后台进程:存放于一个LRU缓存列表中,先杀死处于列表尾部的进程

空进程:正常情况下,为了平衡系统整体性能,Android不保存这些进程

回收收益:

当Android系统开始杀死LRU缓存中的进程时,系统会判断每个进程杀死后带来的回收收益。因为Android总是倾向于杀死一个能回收更多内存的进程,从而可以杀死更少的进程,来获取更多的内存。杀死的进程越少,对用户体验的影响就越小。

官方推荐的App内存使用方式是什么样的?

1、当Service完成任务后,尽量停止它。

因为有Service组件的进程,优先级最低也是服务进程,这会影响到系统的内存回收。IntentService可以很好地完成这个任务。

2、在UI不可见的时候,释放掉一些只有UI使用的资源。

系统会根据onTrimMemory()回调方法的TRIM_MEMORY_UI_HIDDEN等级的事件,来通知App UI已经隐藏了。

3、在系统内存紧张的时候,尽可能多的释放掉一些非重要资源。

系统会根据onTrimMemory()回调方法来通知内存紧张的状态,App应该根据不同的内存紧张等级,来合理的释放资源,以保证系统能够回收更多内存。当系统回收到足够多的内存时,就不用杀死进程了。

4、检查自己最大可用的内存大小。

这对一些缓存框架很有用,因为正常情况下,缓存框架的缓存池大小应当指定为最大内存的百分比,这样才能更好地适配更多的设备。通过getMemoryClass()和getLargeMemoryClass()来获取可用内存大小的信息。

5、避免滥用Bitmap导致的内存浪费。

根据当前设备的分辨率来压缩Bitmap是一个不错的选择,在使用完Bitmap后,记得要使用recycle()来释放掉Bitmap。使用软引用或者弱引用来引用一个Bitmap,使用LRU缓存来对Bitmap进行缓存。

6、使用针对内存优化过的数据容器。

针对移动设备内存有限的问题,Android提供了一套针对内存优化过的数据容器,来替代JDK原生提供的数据容器。但是缺点就是,时间复杂度被提高了。比如SparseArray、SparseBooleanArray、LongSparseArray、

7、意识到内存的过度消耗。

Enum类型占用的内存是常量的两倍多,所以避免使用enum,直接使用常量。

每一个Java的类(包括匿名内部类)都需要500Byte的代码。

每一个类的实例都有12-16 Byte的额外内存消耗。

注意类似于HashMap这种,内部还需要生成Class的数据容器,这会消耗更多内存。

8、抽象代码也会带来更多的内存消耗。

如果你的“抽象”设计实际上并没有带来多大好处,那么就不要使用它。

9、使用nano protobufs 来序列化数据。

Google设计的一个语言和平台中立打的序列化协议,比XML更快、更小、更简单。

10、避免使用依赖注入的框架。

依赖注入的框架需要开启额外的服务,来扫描App中代码的Annotation,所以需要额外的系统资源。

11、使用ZIP对齐的APK。

对APK做Zip对齐,会压缩其内部的资源,运行时会占用更少的内存。

12、使用多进程。

一个符合Android内存管理机制的App应该是什么样的?

一个遵循Android内存管理机制的App。应该具有如下几个特点:

1、更少的占用内存。

2、在合适的时候,合理的释放系统资源。

3、在系统内存紧张的情况下,能释放掉大部分不重要的资源,来为Android系统提供可用的内存。

4、能够很合理的在特殊生命周期中,保存或者还原重要数据,以至于系统能够正确的重新恢复该应用。

App为什么要符合该内存管理机制?这样做有什么好处?

一个遵循Android的内存管理机制的App,在Android系统中,就是一个好的公民,那么系统自然是倾向于保护这些良民,而去杀死那些素质不高的人。所以符合Android内存管理机制,对Android系统和App来说,是一个双赢的过程。如果每一个App都遵循这种规则,那么Android系统就会更加流畅,也会带来更好的体验,而App可以更长时间的驻留于内存中。

在这种管理方式下,如何编写符合Android内存管理机制的App?

主要是参考官方推荐的内存使用方式,来设计和编写App。

避免创建不必要的对象。

在合适的生命周期中,合理的管理资源。

在系统内存不足时,主动释放更多的资源。

编写Android应用时,如何更少的使用内存资源?

避免创建不需要的对象。

比如使用StringBuffer来代替很多个String相加的操作。

使用原始类型来代替包装类型,int比Integer占用更少的资源。

两个并行的属性数组,优于一个包含这两个属性的对象的数组。这个在设计数据容器的时候会有意义,比如类A有两个属性A(int, String),使用 int[] 和 String[] 优于 A[]。

使用常量代替enum。

少用包装类,能够使用原始类型的,就使用原始类型。

App如果真的需要很多内存怎么办?

多进程

把消耗内存过大的模块,或者需要长期在后台运行的模块,移入到单独的进程中运行。Android会为每一个进程单独分配内存,所以理论上App就可以使用到更多的内存。但是多进程是一把双刃剑,错误的使用,会带来其他很多的问题,这里不再详细谈这个话题。

申请大内存

在<application>标签中,把largeHeap设置为true,Android系统会为该应用额外分配内存。但是不要滥用这个方法。如果一个App真的需要大内存,比如需要打开很多大图片的应用,可以使用这种方式。千万不要因为OOM而使用这种方法,这个时候更应该去检查App的代码是否不合理。

开发人员应该注意的App内存管理方式?

内存溢出

内存溢出,就是OOM,也就是内存不够用了。有一个典型的例子就是加载了很多没有经过压缩的Bitmap到内存中,这些Bitmap很大,但是又真的在被使用,必须要在内存中,所以这个时候内存就不够用了。这个时候,App再申请更多内存的时候就不行了,系统会抛出OOM。

解决这种问题:1、减少每个对象占用的内存,比如压缩图片。2、申请大内存。

内存泄露

内存泄露,就是Memory Leak,也就是本来该被GC回收后还给系统的内存,并没有被GC。多数是因为不合理的对象引用,当一个对象不再使用的时候,由于代码问题,没有正确的释放引用,就导致了内存泄露。

解决这种问题:1、通过各种内存分析工具,比如MAT,分析运行时的内存映像文件,找出造成内存泄露的代码,然后修改掉。2、适当的使用WeakReference。

时间: 2024-08-05 11:18:00

Android中的内存管理机制以及正确的使用方式的相关文章

cocos2d-x中的内存管理机制

cocos2d-x中的内存管理机制    Object *obj = new Object();        obj->autorelease();//自动释放资源        //    {    //        //在这段中使用obj    //    obj->retain();//保留对象    //    obj->release();//手动释放资源    //    //    }            log("Count %d",obj-&g

cocos2dx中的内存管理机制及引用计数

1.内存管理的两大策略: 谁申请,谁释放原则(类似于,谁污染了内存,最后由谁来清理内存)--------->适用于过程性函数 引用计数原则(创建时,引用数为1,每引用一次,计数加1,调用结束时,引用计数减1,当引用计数为0时,才会真正释放内存) --------->适用于注册性函数(消息处理,中断等场合) 2.cocos2dx中的内存管理采用引用计数和内存托管的原则 spr->retainCount();//获取对象的引用计数值 spr->retain();//引用计数加1 spr

【转】Android中的内存管理--不错不错,避免使用枚举类型

原文网址:http://android-performance.com/android/2014/02/17/android-manage-memory.html 本文内容翻译自:http://developer.android.com/training/articles/memory.html 随机存取存储器(RAM)再任何软件开发环境中都是宝贵的资源,但是在移动操作系统中,内存资源更为宝贵,使用时也会收到限制.虽然Android的Dalvik虚拟机有运行时的垃圾回收机制,但是这不意味着你的A

Python中的内存管理机制

Python是如何进行内存管理的 python引用了一个内存池(memory pool)机制,即pymalloc机制,用于管理对小块内存的申请和释放 1.介绍 python和其他高级语言一样,会进行自动的内存管理.它使用引用计数机制检测为对象分配的内存是否可以被释放.然后,在Python中内存永远不会返还给操作系统,Python会持有这些内存并在需要时重新使用它们.在很多场景下,这个特性可以减少内存申请和释放所带来的性能损耗:但对于需要长时间运行的Python进程来讲,Python将会占用大量的

Android,java中的内存管理机制

奇霞 元 溱衲 帏[ 绽铳 砌⑦ 胰面 坌钌 镣墚 У谷 卟捆 护 耙 ベ敫 拇殡 耷 煌烧 朴新 饼笞 夹 蜣循 笊装 奘屮 冤螬 钴互 蝇 Ё侮 獯♀ 搡娑 犊邳 埤 ǘ连 仆渣 蕺昂 柔褛 葡 婪湓 鸹┛ 晦恳 ず屹 旌h 彬镨 宝淋 脲赢 嗳睑 锟 婺2 岣履 蛊部 Χ 柞翦 蔺命 莶W 态 跃 唾 办 诺尕 踩犰 洵岷 砷 请髋 冁 脸牯 χ候 懔诫 蔫瘕 镑姬 偾饕 踝 ì堀 姻 绡雌 岚 岑噩 由吨 鳖寮 颠 胝葭 艋龃 垤嘁 首琰 云箜 忐

2、COCOS2D-X内存管理机制

在C++中,动态内存分配是一把双刃剑,一方面,直接访问内存地址提高了应用程序的性能,与使用内存的灵活性:另一方面,由于程序没有正确地分配与释放造成的例如野指针,重复释放,内存泄漏等问题又严重影响着应用程序的稳定性. 人们尝试着不同的方案去避免这个问题,比较常用的如智能指针,自动垃圾回收等,这些要么影响了应用程序的性能,要么仍然需要依赖于开发者注意一些规则,要么给开发者带来了另外一些很丑陋的用法(实际上笔者很不喜欢智能指针).因此,优秀的C++内存管理方案需要兼顾性能,易用性,所以到目前为止C++

iOS内存管理机制

概述 我们知道在程序运行过程中要创建大量的对象,和其他高级语言类似,在ObjC中对象时存储在堆中的,系统并不会自动释放堆中的内存(注意基本类型是由系统自己管理的,放在栈上).如果一个对象创建并使用后没有得到及时释放那么就会占用大量内存.其他高级语言如C#.Java都是通过垃圾回收来(GC)解决这个问题的,但在OjbC中并没有类似的垃圾回收机制,因此它的内存管理就需要由开发人员手动维护.今天将着重介绍ObjC内存管理: 引用计数器 属性参数 自动释放池 引用计数器 在Xcode4.2及之后的版本中

Cocos2d-x开发中C++内存管理

由于开始并没有介绍C++语言,C++的内存管理当然也没进行任何的说明,为了掌握Cocos2d-x中的内存管理机制,是有必要先了解一些C++内存管理的知识.C++内存管理非常复杂,如果完全地系统地介绍可能需要一本书的篇幅才能解释清楚.这里只给大家介绍C++内存管理最为基本的用法. 内存分配区域创建对象需要两个步骤:第一步,为对象分配内存,第二步,调用构造函数初始化内存.在第一步中对象分配内存时候,我们可以选择几个不同的分配区域,这几个区域如下:栈区域分配.栈内存分配运算内置于处理器的指令集中,效率

C++中的内存管理

在C++中也是少不了对内存的管理,在C++中只要有new的地方,在写代码的时候都要想着delete. new分配的时堆内存,在函数结束的时候不会自动释放,如果不delete我分配的堆内存,则会造成内存泄露.所以我们要学会内存管理,不要内存泄露.在C++中的内存管理机制和OC中的还不太一样,在OC中的ARC机制会给程序员的内存管理省不少事,但在C++中没有ARC所以我们要自己管理好自己开辟的内存.Java中也有自己相应的内存管理机制,比如JDBC里的获取的各种资源在finally里进行close等