安卓怎么清理内存(Android优化内存优化)
安卓怎么清理内存(Android优化———内存优化)Java虚拟机Java内存模型
虚拟机栈(线程私有):局部变量表、操作数栈、动态链接、方法出口等信息
堆(线程共享):实例对象
方法区(线程共享):类信息,常量,即时编译器编译后的代码
程序计数器(线程私有):字节码行号指示器,记录当前线程执行到多少行
本地方法栈(线程私有):和虚拟机栈类似,两者的区别就是虚拟机栈是为虚拟机执行java方法服务,本地方法栈为虚拟机执行native方法服务 。程序计数器
线程计数器中如果正在执行java方法,计数器记录的是当前指令的地址,
如果是Native方法,计数器记录为空堆
堆内存 = 新生代(1) + 老年代(2)
新生代:复制算法
老年代:标记整理算法方法区
也叫"永久代",1.8以后将方法区去除了,将方法区移动到直接内存
内存回收主要考虑堆区和方法区的回收,其他部分会根据线程的产生和消亡个版本区别
1.6:运行常量池在方法区
1.7:运行常量池在堆中
1.8:删除方法区,引入直接内存,元空间概念,方法区中的静态变量被转移到堆中,只有class元数据在元空间。
堆中的老年代和方法区(永久代)是绑定的,无论哪一方满了,都会触发双方的GC回收问题:
堆和栈的区别:栈:基本数据类型变量(int、short、long、byte、float、double、boolean、char)以及对象的引用变量堆:存储java对象堆中的对象对所有线程可见,栈内存只属于一个线程堆的内存空间远远大于栈
为什么删除方法区?启动大小固定,很难调优,容易发生OOM元空间在本地内存中分配,本地内存足够就不会溢出GC垃圾回收判断对象是否存活
引用计数算法(缺点:循环引用,技数永远不为0)
可达性算法(二叉树中向下搜索,不存在引用链则对象不可用)回收算法
标记清除算法:标记完后对对象进行回收,使用在老年代缺点:效率不高,标记和清除效率不高差生大量碎片空间,导致空间浪费
复制算法:将可用对象复制到新的连续空间,删除之前的空间缺点:浪费50%的内存,复制长生存期的对象效率低下,所以该算法使用在新生代
标记整理算法:前期使用标记清除算法,后续使用整理算法,使对象排列称联系空间,使用在老年代
分代收集算法:对数据进行分代,每一代执行不同的回收算法
年轻代分为eden、s0、s1区,分别为8:1:1,年轻代和老年代为1:2
元空间的gc:元空间中的类加载器存活,则元空间中元数据也存活
Minor GC : 清理年轻代
Major GC : 清理老年代
Full GC : 清理整个堆空间,包括年轻代和永久代四大引用介绍简述
强引用:Strong Reference,通常使用的对象方式,gc不会回收
软引用:SoftReference,当内存不足时进行回收
弱引用:WeakReference,下一次gc时回收
虚引用:PhantomReference,任何时候可回收
在内存泄露问题处理上,使用最多的是弱引用,许多源码、框架都是用
eg:
ThreadLocalMap中存储以ThreadLocal的弱引用为键,具体内容为value
Glide中缓存使用activeResource,存储的是图片的弱引用
解决Handler的内存泄漏使用弱引用Reference理解
所有的引用都是继承自Reference,以下以WeakReference为例:publicclassWeakReference<T>extendsReference<T>{/** *Createsanewweakreferencethatreferstothegivenobject.Thenew *referenceisnotregisteredwithanyqueue. * *@paramreferentobjectthenewweakreferencewillreferto */ publicWeakReference(Treferent){super(referent); }/** *Createsanewweakreferencethatreferstothegivenobjectandis *registeredwiththegivenqueue. * *@paramreferentobjectthenewweakreferencewillreferto *@paramqthequeuewithwhichthereferenceistoberegistered, *or<tt>null</tt>ifregistrationisnotrequired */ publicWeakReference(Treferent,ReferenceQueue<?superT>q){super(referent,q); } }
其中存在两种构造方法,区别在于是否传入引用队列,如果不传入引用队列,说明只存在一种引用,不需要引用队列成链存储publicabstractclassReference<T>{privatestaticbooleandisableIntrinsic=false;privatestaticbooleanslowPathEnabled=false;//引用的对象,由垃圾回收器控制其引用 volatileTreferent;/*TreatedspeciallybyGC*/ finalReferenceQueue<?superT>queue; ReferencequeueNext; Reference<?>pendingNext;publicTget(){returngetReferent(); }@FastNative privatefinalnativeTgetReferent();publicvoidclear(){ clearReferent(); }@FastNative nativevoidclearReferent();publicbooleanisEnqueued(){//Contrarytowhatthedocumentationsays,thismethodreturnsfalse //afterthisreferenceobjecthasbeenremovedfromitsqueue //(b/26647823).ReferenceQueue.isEnqueuedpreservesthishistorically //incorrectbehavior. returnqueue!=null&&queue.isEnqueued(this); }publicbooleanenqueue(){returnqueue!=null&&queue.enqueue(this); }/*--Constructors--*/ Reference(Treferent){this(referent,null); } Reference(Treferent,ReferenceQueue<?superT>queue){this.referent=referent;this.queue=queue; } }
抽象类很简短,可以看出一个关键点,Reference是一个节点,保存next的引用,方法调用都是使用ReferenceQueue方法,直接进入:privateReference<?extendsT>head=null;privateReference<?extendsT>tail=null;booleanenqueue(Reference<?extendsT>reference){ synchronized(lock){if(enqueueLocked(reference)){ lock.notifyAll();returntrue; }returnfalse; } }privatebooleanenqueueLocked(Reference<?extendsT>r){ ...if(rinstanceofCleaner){ Cleanercl=(sun.misc.Cleaner)r; cl.clean(); r.queueNext=sQueueNextUnenqueued;returntrue; }if(tail==null){ head=r; }else{ tail.queueNext=r; } tail=r; tail.queueNext=r;returntrue; }
入队方法中,
使用synchronized添加锁,入队结束后释放锁,在ReferenceQueue中并不是标准的队列,使用的是Reference节点成链,行成单链表,类似于MessageQueue.
如果是Cleaner类,创建一个虚引用节点,即不如队。Cleaner是用来释放非堆内存,所以做特殊处理
SoftReference>publicclassSoftReference<T>extendsReference<T>{//时间戳,由gc更新 staticprivatelongclock;privatelongtimestamp;publicSoftReference(Treferent){super(referent);this.timestamp=clock; }/** *Createsanewsoftreferencethatreferstothegivenobjectandis *registeredwiththegivenqueue. * *@paramreferentobjectthenewsoftreferencewillreferto *@paramqthequeuewithwhichthereferenceistoberegistered, *or<tt>null</tt>ifregistrationisnotrequired * */ publicSoftReference(Treferent,ReferenceQueue<?superT>q){super(referent,q);this.timestamp=clock; }publicTget(){ To=super.get();if(o!=null&&this.timestamp!=clock)this.timestamp=clock;returno; } }
由gc管理时间戳
clock:上一次gc时间
timestamp:访问get时最近一次的gc时间
回收条件为:clock - timestamp <= free_heap * ms_per_mb
free_heep为堆空间空闲大小
ms_per_mb是保留软引用时间/MB
PhantomReferencepublicclassPhantomReference<T>extendsReference<T>{publicTget(){returnnull; }publicPhantomReference(Treferent,ReferenceQueue<?superT>q){super(referent,q); } }
虚引用的get方法返回null,不做gc保留
虚引用通过构造方法可以查看是持有对象引用的
总结:所有引用都是继承自Reference基类的,该类是一个链表节点,ReferenceQueue通过这点形成单链表,称之为队列,进行引用管理,所有引用都可以通过Reference的isEnqueue方法判断引用是否存在。FinalizerReference理解
java堆中创建对象时,如果java类定义了finalize方法,就会新建一个FinalizerReference类,指向这个新建的对象内存问题
内存泄漏:内存没有按照预期在gc时回收
内存溢出:内存大小超出指定大小,导致OOM
内存抖动:短时间创建大量内存对象,然后回收,导致内存发生锯齿形抖动,内存空间不连续加上碎片会导致更大的空间,最终OOM内存优化意义
减少OOM,提高系统稳定性
减少卡顿,提高流畅度
减少内存占用,提高应用存活率
减少异常发生和代码逻辑隐患Android内存泄漏常见内存泄漏
匿名内部类持有外部类引用,导致外部类内存泄漏(Handler)
单例传入Context导致调用单例方无法被回收。
非静态内部类创建静态实例
注册与反注册
资源对象关闭
集和及时清理内存泄漏检测
Profiler,Memory Analyzer(MAT)
Android studio自带内存、cpu、网络的变化,可以根据内存变化做具体分析
LeakCanary
框架集成,自动检测内存泄漏,生成app,提供内存泄漏栈堆情况
原理:绑定生命周期,对Activity和Fragment来说,在onDestory时将对象放入弱引用队列进行存储,触发gc后,如果还存在,则发生内存泄漏
StrictMode(很少用)
一款比较老的工具,ThreadPolicy可以检测主线程是否网络访问,是否读写。VMPolicy检测内存,Activity,Fragment是否泄漏,资源是否正确关闭内存优化空间
不必要的自动装箱
自动装箱就是将基础数据类型转化为对应的复杂类型,在HashMap的增删改查中充满了自动装箱问题,所以尽量避免这中问题,如将HashMap替换为SparseArray和ArrayMap
内存复用
资源复用:通用字符串,颜色,布局
视图复用:类似于RecyclerView的优化复用
对象池:创建对象池,不用重复创建对象,类似于线程池,messae享元模式
Bitmap对象复用:使用inBitmap属性可以告知Bitmap解码器尝试使用已经存在的内存区域,新解码的bitmap会尝试使用之前那张bitmap在heap中占据的pixel data内存区域。
在App可用内存过低时主动释放内存在App退到后台内存紧张即将被Kill掉时选择重写Application中 onTrimMemory/onLowMemory 方法去释放掉图片缓存、静态缓存来自保。
其他场景优化item被回收不可见时释放掉对图片的引用ListView:因此每次item被回收后再次利用都会重新绑定数据,只需在ImageView onDetachFromWindow的时候释放掉图片引用即可。RecyclerView:因为被回收不可见时第一选择是放进mCacheView中,这里item被复用并不会只需bindViewHolder来重新绑定数据,只有被回收进mRecyclePool中后拿出来复用才会重新绑定数据,因此重写Recycler.Adapter中的onViewRecycled()方法来使item被回收进RecyclePool的时候去释放图片引用。如果使用字符串拼接,尽量使用StringBuilder、StringBuffer(内存抖动)自定义view减少onDraw的耗时和执行次数尽量使用静态内部类尽量使用基础数据类型合适的时候使用软/弱引用线上监控方案
常规监测当内存使用超过80%,使用Debug.dumpHprofData(String fileName)
获取dump文件回传至服务器,而后手动分析LeakCanary集成并带到线上
Probe线上监测工具
LeakInspector
ResourceCanary最后
小编在网上收集了一些Android 开发相关的学习文档、面试题、Android 核心笔记等等文档,希望能帮助到大家学习提升,如有需要参考的可以直接私信我领取哦
生活知识科普淡水珍珠和海水珍珠的区别在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于淡水珍珠和海水珍珠的区别的这些知识点,这就说明了现在人们也开开渐渐关注起了淡水珍
五行缺火(五行缺火的人怎么补火)五行缺火(五行缺火的人怎么补火)天地乾坤,阴阳五行,金木土水火,五行相生相克生生不息。有的人在出生的时候就注定五行中会缺什么。比如你缺火,人出生时的生辰八字就决定了一生的五行归属,
如何定位孩子手机位置(如何定位孩子手机)如何定位孩子手机位置(如何定位孩子手机)老公出去应酬,酩酊大醉后,倒在了哪条马路上?爸妈年龄大了,我在上班,他们遛弯回没回来?儿子毕业了,一个人去远行,火车现在开到了哪里?生活中有
怎么定位小孩手机位置(怎么定位小孩)怎么定位小孩手机位置(怎么定位小孩)对于一位出门在外,有导航依赖症的小可爱来说,手机定位是多么的重要。一旦发现自己的实际位置和手机定位位置并不一致,内心瞬间崩溃我多想见到你,怎奈手
怎么定位小孩手机位置(怎么定位小孩)怎么定位小孩手机位置(怎么定位小孩)在日常生活中,由于手机的功能越来越强大,所以大部分人手上都会有智能手机。那么你用了这么久的手机,你知道手机上面隐藏的定位功能吗?你知道手机也能定
lucky是什么意思(给孩子取名CandyLucky)lucky是什么意思(给孩子取名CandyLucky)现如今,给孩子取一个英文名已经不是什么新奇的事情了。更有家长为了让孩子学好外语,从幼儿园开始就把孩子送到了双语学校,因此给孩子
幼儿讲故事(讲故事练口才小处不可随便)幼儿讲故事(讲故事练口才小处不可随便)每天一个口才故事,慢慢你就会发现改变了自己,也会发现不再像以前寡言少语。小处不可随便于右任是国民党元老,他精通书法,尤其善于草书。因此,有很多
八十年代电影演员(80年代武打片男星今昔对比照)八十年代电影演员(80年代武打片男星今昔对比照)昨天,我又把少林寺翻出来看了一遍,其中的每一个镜头,每一帧画面,都能唤起我们的太多回忆。当看到李连杰胡坚强等人一一出现在影片中,我不
怎么给自己的手机定位(如何精准定位车辆位置)怎么给自己的手机定位(如何精准定位车辆位置)如何精准定位车辆位置?相信很多朋友一定和小编遇到过一样的问题,特别是带有路痴属性的女同学,今天,就把解决这个问题的方案分享给大家。正确选
什么软件能定位车辆位置(什么软件能定位车辆)什么软件能定位车辆位置(什么软件能定位车辆)曾有这么一个说法,当手机电量不足时,按3370键,手机里的备用电池就会自动启用。有许多人在电池快没了,而情况又不允许充电时,都尝试过,发
什么软件能定位车辆位置(什么软件能定位车辆)什么软件能定位车辆位置(什么软件能定位车辆)手机消费成为当今最常见的支付手段,手机丢失怎么办?别慌!今天来今天来教大家一招,如何快速定位手机位置!以小米手机为例1找到手机设置,打开
生活知识科普出伏是不是就凉快了在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于出伏是不是就凉快了的这些知识点,这就说明了现在人们也开开渐渐关注起了出伏是不是就
生活知识科普牛肉不能和什么一起吃会食中毒在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于牛肉不能和什么一起吃会食中毒的这些知识点,这就说明了现在人们也开开渐渐关注起了牛
过了冬至白昼是不是越来越长(冬至后白天越来越长吗天气有什么变化)过了冬至白昼是不是越来越长(冬至后白天越来越长吗天气有什么变化)冬至后白天越来越长吗天气有什么变化冬至养生知识吃什么12月21日下午6时02分,将迎来冬至节气。这也是北半球一年中白
生活知识科普切开后的西瓜吃不完怎么办在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于切开后的西瓜吃不完怎么办的这些知识点,这就说明了现在人们也开开渐渐关注起了切开后
生活知识科普点痣后不能吃什么在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于点痣后不能吃什么的这些知识点,这就说明了现在人们也开开渐渐关注起了点痣后不能吃什
生活知识科普漂唇后能吃韭菜吗在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于漂唇后能吃韭菜吗的这些知识点,这就说明了现在人们也开开渐渐关注起了漂唇后能吃韭菜
生活知识科普漂唇后能吃鸡肉吗在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于漂唇后能吃鸡肉吗的这些知识点,这就说明了现在人们也开开渐渐关注起了漂唇后能吃鸡肉
生活知识科普漂唇后能吃鸡蛋吗在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于漂唇后能吃鸡蛋吗的这些知识点,这就说明了现在人们也开开渐渐关注起了漂唇后能吃鸡蛋
生活知识科普漂唇后能吃牛肉吗在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于漂唇后能吃牛肉吗的这些知识点,这就说明了现在人们也开开渐渐关注起了漂唇后能吃牛肉
生活知识科普花露水对蟑螂有效吗在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于花露水对蟑螂有效吗的这些知识点,这就说明了现在人们也开开渐渐关注起了花露水对蟑螂
生活知识科普穿什么颜色容易被蚊子叮在我们的生活当中有很多的常识性的知识大多数人都是不知道的,就好比最近就有很多小伙伴咨询小编问小编关于穿什么颜色容易被蚊子叮的这些知识点,这就说明了现在人们也开开渐渐关注起了穿什么颜