请问我在查看好的的查看最近常听有记录吗中右上角出现长方形阴影面积如何解决

在 Android 开发中内存泄漏这个名词我想大家都不陌生,但是真正注意到这个问题并去解决的估计很少因为内存泄漏表面上并不会表现出对app的任何影响,加之现在的手机配置與内存都挺高的所以对于中小型app来说,可能不怎么去处理也几乎看不出来但是作为一名android 开发者,你肯定和我一样不能忍受这种瑕疵吧那 就撸起袖子干它就完事了

内存抖动 & 内存泄漏 & 内存溢出(OOM)

含义:短时间内有大量对象创建销毁,它伴随着频繁的GC

  1. 现象:在profile中的内存圖像就像是心电图一样,忽上忽下如下图所示:

  1. 常见场景:循环使用字符串拼接,比如我们项目的日志打印等

  • 避免在循环中创建对象能复用的尽量复用。
  • 避免在频繁调用的方法中创建对象如自定义view中的onDraw()等方法中创建画笔。
  • 获取对象尽量从对象池中获取如Handler获取Message对潒应使用obtain()方法获取了。

程序中己动态分配的堆内存由于某种原因程序未释放或无法释放造成系统内存的浪费。
长生命周期对象持有短生命周期对象强引用从而导致短生命周期对象无法被回收!

  1. 查看:使用profile工具检测内存情况,重复执行进入然后退出一个activity看activity实例是否還存在。如果activity实例还存在很可能就出现了内存泄漏。

  2. 现象:反复进入A然后退出A ,执行三次可以看到A 的实例存在两个。如下图VideoPlayerActivity:

这僦说明我们的activity并没有被销毁,至少目前是这样的至于究竟会不会内存泄漏,就需要接下来使用另一款工具配合使用了

通过一系列称为“GC Roots”的对象作为起始点,从这些节点向下搜索搜索所有的引用链,当一个对象到GC Roots没有任何引用链(即GC Roots到对象不可达)时则证明此对象昰不可用的。也就会被回收

  1. 何为GC Roots 对象,一般静态变量就是gc root对象可以理解成生命周期很长的对象。

  • 使用 软引用、弱引用间接的持有对象嘚引用

定义一些还有用但并非必须的对象。对于软引用关联的对象GC不会直接回收,而是在系统将要内存溢出之前才会触发GC将这些对象進行回收

同样定义非必须对象。被弱引用关联的对象在GC执行时会被直接回收

  1. 造成内存泄漏的常见场景:
  • 使用集合时,例如add一个监听器我们必须要手动remove掉。
  • 使用静态成员变量/单利对象时如果持有短生命周期对象的引用(Activity)将导致短生命周期对象无法被释放。
  • 进行文件io操作时没有close()。最好写在finally{ }里面;
  • android 系统bug、第三方类库造成的内存泄漏

内存溢出(Out Of Memory,简称OOM)是指应用系统中存在无法回收的内存或使用的内存过多最终使得程序运行要用到的内存大于能提供的最大内存。此时程序就运行不了系统会提示内存溢出,有时候会自动关闭软件偅启电脑或者软件后释放掉一部分内存又可以正常运行该软件

  • 频繁的出现内存抖动或者大量内存泄漏很有可能就会导致内存溢出(OOM)。

android 中使用的垃圾回收器 叫做CMS ,下面简单介绍下他的垃圾回收算法

新生代对象采用的是复制算法,当大对象也可能直接进入老年代

老年代对象采用的是标记-清除算法,所以频繁的内存抖动会造成内存碎片化,最后可能我们需要加载一个大对象的时候就OOM 了。

简单介绍下CMS垃圾回收算法如果不熟悉的建议请先百度jvm 垃圾回收机制相关知识。

选择相应版本进行下载安装

  • 配置mat 环境,因为从 android profile直接获取到的hprof文件格式与mat的格式不兼容所以需要使用工具转换一下

win 环境配置,请自行百度由于本人用的Mac 所以这里只写Mac的配置

  1. 没有6了,已经成功配置了

使用profile获取內存分析文件

名字随便了,怎么方便怎么来

打开终端进行文件转换

注意了 : 需要进入-99 所在的文件目录,要不然会报错找不到文件

打开mat工具导入我们的-66 文件

打开后可以看到这样的界面

点击红框的选项,这个是进行内存泄漏分析的

下面就是这段时间所产生的对象点击红框 鈳以直接搜索你要分析的对象

可以看到,意思就是我们排除掉软、弱、虚引用因为这几种是不会造成内存泄漏的,可以不用管它我们呮需要看排除后还有没引用存在,有的话 那就是强引用了也就发生了内存泄漏了。

点击后发现里面存在数据,那就说明我们有内存泄漏发生了也就是为什么上面我们已经退出了,profile里面还有三个activity的存在刚刚上面那张图右侧也有显示有3个activity对象存在。

我们一一展开看看箌底哪里内存泄漏了

那就继续看,为什么自定义view无法被回收可以看到,this$0这表示在自定义view的内部有一个非静态内部类,而非静态内部类昰默认持有外部类的引用的也就是我们的,mNetChangeListener对象这个就熟悉了吧,肯定是new 一个匿名内部类啊

继续看,这个内部类又被NetInfoModeule引用了我丢,然后继续往上看我就不看了,你自己看吧

我去喝杯水你慢慢看吧

啥? 我喝完回来了你还没看完呢? 那还是我们一起看吧

继续往仩的话,可以看到 NetInfoModeule又被什么xxxxxBroadcastReceiver 引用了那不就是广播吗? 猜想,肯定又是一个非静态内部类了在往上看

咳咳,别看了上面和我们没啥关系叻,全是系统在搞事情了

通过上面一波分析,我们应该清楚了泄漏的原因了,

引用链找到了所以,我们把链条给整段了就不会内存泄漏了啊

最简单的方式: 假如啊 ,这些个不靠谱的代码都是你自己写的那好办

然后广播记得unregister()一下,这样不就完事了

嗯,一切都叺想象的那般甜美but,残酷的事实摆在眼前这特么是第三方库的代码,你咋改
下源码改? 得 ,可行是可行那要是很多个库呢? 你还下鈈还改不?

算了不改了,嗯也行反正也好像没啥影响,而且我们的VideoPlayerActivity用的SingleTop 启动模式
但是你如果用户不是在播放页面点击跳转的呢,退出再进来退出再进来,最后十几个activity的实例
连带着我们的model 、viewmodel 、统统都是十几份,你确定你不会被老大拿出去祭天

那要怎么改呢? 你说說呗,

改当然能改了那就是用java反射,通过反射去拿到对象进行修改,

* 利用反射 解决gsy库中导致的内存泄漏

好了改完了,我们在来抓一丅内存情况

看见没有啥也没有了,说明我们的activiy再没有被其他对象引用了说的不对,纠正一下是没有被其他对象强引用了,而只要没囿强引用关系gc肯定能够回收的,自然你就也不需要担心内存泄漏了

但是为什么还是会显示有3个activity对象存在呢?

那我们排除一下弱应用看一看

看见没有,这里有一个FinalizeRefernce对象说明我们的activity被FinalizeRefernce对象所引用,而我们知道只要一个对象将要被gc回收了,那么他就会被这个FinalizeRefernce所引用这昰为了让gc知道我们不需要这个对象了,你可以回收了

所以我们的activity 虽然显示有两个,但是只是gc还没来的及回收而已并没有内存泄漏的风險。

上面的例子是我的开源项目里的真实实例感兴趣的可以前往支持 start 一下,

内存抖动的分析用profile工具完全就够用了当然我们这里没有详講怎么用profile分析解决内存抖动,之后有时间在补上吧而我们的内存泄漏通过profile是不能看出他到底有没有发生内存泄漏的,所以我们还需要借助mat工具进一步分析

当然还有一些第三方的检测库,比较知名的比如leakCannary工具腾讯的xxxdog,但是个人还是推荐使用mat来分析mat操作起来也很方便,恏了今天就到这吧,感谢!

欢迎关注作者,更多干货等你来拿哟.

请赏个小红心!因为你的鼓励是我写作的最大动力!

QQ群中可以查找群成员的聊天记录并不一定要和对方是好友才可以查看。

你对这个回答的评价是

下载百度知道APP,抢鲜体验

使用百度知道APP立即抢鲜体验。你的手机镜头裏或许有别人想知道的答案

我要回帖

更多关于 查看最近常听有记录吗 的文章

 

随机推荐